项目中遇到把购物车中勾选的商品数据传递到订单确认页面时,我直接将整个购物车商品的数据模型传递过去,然后再在订单确认页面里对模型里面的商品数据进行了删除,而这样的操作也会影响到购物车页面,所以需要进行拷贝操作。虽然不算太难的解决了这个问题,但是感觉时误打误撞,所以翻看了《Objective-C程序设计》有了这篇记录。
区分深拷贝和浅拷贝
我们以一个自定义对象的copy操作,来进行解释,在此之前,我们需要了解,如果想要实现自定义对象的copy操作,该类型必须遵循<NSCopying>协议,并且实现 - (id)copyWithZone:(NSZone *)zone
方法。
当实现这个方法时,唯一需要注意是,生成对象副本的语句要如下:MyObject *copyObject = [self class] allocWithZone:zone];
,那么为什么不是这样写呢?MyObject *copyObject = [MyObject allocWithZone:zone]
;因为当该类型的子类SubObject也要实现copy操作,会首先调用父类的copy实现方法,这时就会返回一个新的MyObject类型的对象,而不是SubObject类型,在操作Subject的属性赋值方法里,则会因为MyObject根本没有这个属性,而导致程序崩溃。
非容器对象中的深拷贝和浅拷贝
如果你有一个对象MyPerson *person = [MyPerson alloc] init];
,该对象还有一个name属性person.name = @"testName;
,我们在此对person对象进行copy操作MyPerson *anotherPerson = [person copy]
,浅拷贝的含义是产生新的一个anotherPerson对象,但是其属性还是person一样,person.name指向同一个对象;而深拷贝的含义则是不仅产生新的一个anotherPerson对象,而且其属性anotherPerson.name所指向的对象也是区别于person.name。
浅拷贝:
深拷贝:
首先自定义深拷贝的实现方法
效果如下:
总结:当我们执行copy操作的时候,接受者会产生一个新的对象,但是如果没有自定义深拷贝的实现的话,新的对象和旧的对象的属性都是指向同一个内存地址。
容器对象中的深拷贝和浅拷贝
在OC中,所有Foundation中的容器类,分为可变容器和不可变容器,它们的拷贝都是浅拷贝。对容器进行拷贝时,会产生一个新的容器对象,并且复制容器里的指针。
那么怎么对容器进行深拷贝呢?
我们可以利用方法:
- (instancetype)initWithArray:(NSArray<ObjectType> *)array copyItems:(BOOL)flag;
,flag值为YES
时,会对容器类的对象执行copy操作,注意这里,如果对象是可变对象,那么执行copy操作后,得到的将是不可变对象。
copy 和 mutableCopy
使用copy 或者是 mutableCopy 的时候,不对接受者是可变还是不可变进行要求的,可变对象执行copy
操作后,返回不可变对象,不可变对象执行mutableCopy
操作后,返回可变对象。
如何更容易的去理解深浅拷贝的概念
首先要明白两点:引用计数和指针,当我们进行普通的赋值操作时:
NSMutableString *str1 = [NSMutableString stringWithString:@"test"];
NSMutableString *str2 = str1;
[str2 appendString:@"-test"];
NSLog(@"str1 = %@", str1);
NSLog(@"str2 = %@", str2);
两个指针对象str1
和str2
指向同一处内存地址,而进行拷贝的时候,无论是自定义对象还是容器类,浅拷贝时,copy
得到的自定义对象和原对象的属性指针也是指向同一个内存地址,只是属性指针所指向的对象的引用计数+1而已。