前要
copy这个属性,相信我们用的很多的吧,尤其集合类用的比较多,我们一般都会用到就是NSArray,NSMutableArray,NSString,NSMutableString等等,copy分为深拷贝和浅拷贝,深拷贝就是拷贝里面的内容(完全新的对象),浅拷贝只是拷贝对象地址(一样地址)。@property (copy, nonatomic) NSMutableArray *tempArray;一般都是这么写,然后使用懒加载的方式给这个实例变量进行初始化,但是我使用set方法直接赋值self.tempArray = [NSMutableArray array];然后数组添加元素的时候,就会出现crash。

原因分析
首先既然添加数组的时候crash,会不会是数组没有初始化或者数组不是可变数组的问题呢,以下是我的截图:

可以看出,执行了self.tempArray = [NSMutableArray array];之后,数组变成了不可变的数组,我们添加元素的话肯定有问题了呀,肯定会crash。肯定很多的why???我用的可变数组初始化的呀,为什么变成了不可变数组。其实我们换一种方式来解读一下

我们看出来了吧,我只是把[NSMutableArray array]执行了copy操作,但是one却变成了不可变数组了,那么我们再来看看mutableCopy会是什么结果。

是个可变数组。
再回头看tempArray的属性,是copy属性,这就必然会执行copy操作,然后重新生成了一个新的对象赋值给tempArray,又因为OC是动态语言,只有在执行的时候才知道是什么类型的对象,为什么懒加载就可以避免了这种crash。

– (NSMutableArray *)tempArray{

if (_tempArray) {
_tempArray = [NSMutableArray array];
}
return _tempArray;
}

我们看出来使用的tempArray的实例变量,实例变量是系统帮你生成的,不会存在属性那样的copy的属性,所以不会直接执行copy操作,把可变数组变成不可变数组,实例变量的自然也是可变的。

总结
copy证明可以修饰可变数组,但是会执行copy方法,生成的是个不可变的数组,所以只需要使用实例变量就可以完美解决了,也是常说的使用懒加载的模式来初始化数组对象,其实证明用retain和strong修饰的话,self.tempArray = [NSMutableArray array];这个方法生成的对象也是可变数组。retain引用计数+1;strong也就是强引用一次