Copy基础理解(一)

关于oc深浅拷贝问题可能做过ios开发的都有了解到,尤其是面试特别喜欢问深浅拷贝的问题.但大多有点误区,即便是有几年开发经验的ios程序员来说,很少有人说到伪拷贝这个概念,而只分深浅拷贝,新开了内存就说深拷贝,只拷贝的引用就说浅拷贝,这其实是片面的理解,在这里我们一起来探究一下拷贝究竟怎么区分.
首先创建一个工程,新建一个Person类,给Person类写一个name属性,我们让Person类拥有可拷贝的特性,首先签订NSCoping协议,然后实现copyWithZone方法.代码如下:

  @interface Person:NSObject<NSCopying>
  @property(nonatomic,retain)NSSting *name;
  @end

1.伪拷贝--拷贝地址,相当于retain操作,造成对象引用计数+1
在copyWithZone中代码如下

  - (id)copyWithZone:(NSZone *)zone
  {
    / / 伪拷贝
    return [self retain];
  }

2.main.m文件中创建Person对象并且拷贝一个对象出来,分别打印其地址以及name属性地址,以p1,p2的引用计数.

    Person *p1 = [[Person alloc] init];
    p1.name = @"测试";
    // 执行拷贝
    Person *p2 = [p1 copy];
    // 输出地址查看
    NSLog(@"%p %p %p %p",p1,p2,p1.name,p2.name);
    // 查看 p1 p2引用计数器
    NSLog(@"%ld %ld",[p1 retainCount],[p2 retainCount]);

以发现p1,p2地址相同,两个name地址也相同,而引用计数为2.可见这种拷贝仅仅拷贝了一个引用然后对对象的引用计数+1,跟retain并无分别.

2.浅拷贝--对象开辟新的空间,但是对象的实例变量指向同一块空间
copyWithZone方法中代码如下:

  - (id)copyWithZone:(NSZone *)zone
  {
      // 浅拷贝
      Person *person1 = [Person allocWithZone:zone];
      person1.name = self.name;
      return person1;
  }

3,深拷贝--对象开辟新的空间,两个对象的实例变量也指向不同的空间。
copyWithZone方法中代码如下:
- (id)copyWithZone:(NSZone *)zone
{
// 深拷贝
Person *person2 = [Person allocWithZone:zone];
person2.name = [self.name mutableCopy];
return person2;
}
运行程序查看打印的地址信息发现,p1,p2地址不同,p1.name,p2.name地址不同,p1,p2引用计数都为1,说明现在p1,p2再没有一点瓜葛,完全是在内存中独立存在,也就是开辟新的空间,并且实例变量也指向不用的空间,这才是真正意义上的深拷贝.

demo 地址:https://github.com/HBG-426/Copy

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容