浅拷贝,深拷贝以及全拷贝

iOS里面深浅拷贝问题已经是老生常谈的问题了,而且也几乎是面试必问的提。我这篇文章说的也不是什么高深技术点,只是个小细节的东西,平时能注意到就行

系统的那些对象如:NSString,NSDictionary,NSArray以及他们的子类等,都可以直接使用copy和mutableCopy,系统已经帮忙给他们实现了,NSCopying,NSMutableCopying协议,但是我们自定的对象如果想要使用copy或mutableCopy,必须手动实现对应的NSCopying或NSMutableCopying协议,并且实现对应的方法:

- (id)copyWithZone:(nullableNSZone*)zone;   和

- (id)mutableCopyWithZone:(nullableNSZone*)zone

浅拷贝:指针复制,复制出来的指针指向的还是同一块区域。通过其中一个指针对内容进行修改,会影响另一个指针,因为他俩其实指向同一块区域。

怎样会发生浅拷贝?

就用系统的NSString,NSDictionary,NSArray来说,这些不可变对象,注意重点,不可变对象,他们直接调用copy(不是mutableCopy)则发生的浅拷贝;

直接上图上代码:


浅拷贝,深拷贝以及全拷贝_第1张图片
浅拷贝,深拷贝以及全拷贝_第2张图片

这里发生的浅拷贝,前后两个数组打印的东西一样,当然这不能证明两者指向的区域一样,因为哪怕是深拷贝,这里的打印结果两者也是一样的。由于是不可变数组,没法用移除数组元素来证明,但是可以肯定的是,如果此时其中一个数组移除(假设可以移除)Student对象,另一个数组肯定也同时移除了。有同学可能就要说了,可不可以通过改变复制出来的数组里的Student对象的属性,来证明是浅拷贝。这种肯定是不行的,因为就算是深拷贝,这里的Student对象也只有一份,这也是这篇文章最重要的一点,关于全拷贝。请听下文分解。

深拷贝:内容复制,其实直接说内容复制是不准确的,会让人误解,以这里的数组为例,会让人以为数组里头的对象也会被复制。

怎样会发生深拷贝?

如果是不可变对象如NSString,NSDictionary,NSArray来说,只有调用mutableCopy才会发生深拷贝,如果是他们的可变子类,则调用copy或mutableCopy都会发生深拷贝;

上图:

不可变对象调用mutableCopy发生深拷贝

浅拷贝,深拷贝以及全拷贝_第3张图片
浅拷贝,深拷贝以及全拷贝_第4张图片

可变对象调用copy发生深拷贝,可变对象调用mutableCopy那也毫无疑问是深拷贝,可自证


浅拷贝,深拷贝以及全拷贝_第5张图片
浅拷贝,深拷贝以及全拷贝_第6张图片

这些都是深拷贝,结果通过图片一目了然;

接下来就是重点了,为什么说“深拷贝是内容复制”这种说法不准确?看图:


浅拷贝,深拷贝以及全拷贝_第7张图片
浅拷贝,深拷贝以及全拷贝_第8张图片

可以看到,即使是深拷贝,数组里头的内容仍然只有一份,修改了一个Student对象的name,所有的Student对象都发生了改变。

那怎样才能实现内容复制呢,即我说的全拷贝。系统提供了方法,这里以数组为例,系统方法:

- (instancetype)initWithArray:(NSArray *)array copyItems:(BOOL)flag


浅拷贝,深拷贝以及全拷贝_第9张图片
浅拷贝,深拷贝以及全拷贝_第10张图片

系统提供了这一类方法,后面带一个是否进行内容拷贝的标识,YES则表示数组里头的内容也被复制。对应的字典也有类似方法。但是要注意,这个时候如果集合里头的对象是自定义对象,比如这里的Student对象,那必须让Student实现NSCopying协议,并实现-(id)copyWithZone:(NSZone*)zone方法。否则会崩。这种拷贝我给取名为全拷贝!

你可能感兴趣的:(浅拷贝,深拷贝以及全拷贝)