博客> 为什么NSMutableArray用copy修饰会crash?
为什么NSMutableArray用copy修饰会crash?
2018-11-15 11:31 评论:0 阅读:752 ios_wj@163.com
ios crash copy NSMutableArray

在网上经常看到这样的温馨提示,创建NSArray 属性时要用copy关键词,而创建NSMutableArray的属性时要使用strong修饰。开始时不大理解,表示自己在项目里NSArray用的也是strong... 先做一片总结归纳,仅作为以后忘记了查看使用

废话不多说,先上代码

@interface ViewController() @property(nonatomic,copy)NSMutableArray*arr1; @end @implementation ViewController

  • (void)viewDidLoad { [superviewDidLoad]; NSMutableArray*array = [NSMutableArrayarrayWithObjects:@1,@2,@3,nil]; self.arr1 = array;
    [self.arr1 removeObject:@1]; } //会crash 报错reason: '-[__NSArrayI removeObject:]: unrecognized selector sent to instance 0x6000000487c0'}

为什么呢?

1,先对copy 和 strong的区别做个分析, 其区别主要体现在对应的setter方法的实现不一样,以示例中的arr1为例

如果用copy 修饰符,对应的setter方法实现如下

-(void)setArr1:(NSMutableArray *)arr1{ if(_arr1 != arr1){
[_arr1release];
_arr1= [arr1copy];//内容拷贝,深拷贝 }}

如果修改成用strong修饰符,对应的setter方法实现如下

  • (void)setArr1:(NSMutableArray*)arr1 { if(_arr1 != arr1) {
    [_arr1 release];
    _arr1 = [arr1retain];//指针拷贝,浅拷贝 }}

然而这仍然解释不了上文的why...再看

2,原来不管是集合类对象(NSArray,NSDictionary,NSSet...),还是非集合类对象(NSString),接收到copy或者mutableCopy消息时,都需遵循以下准则:

copy返回的都是不可变对象,所以如果对copy返回值去调用可变对象的接口就会crash. (好像解答了上述问题)

mutableCopy 返回的都是可变对象

3,为什么NSArray要用copy修饰符?看以下例子

@interface ViewController() @property(nonatomic,copy)NSArraycArr; @property(nonatomic,strong)NSArraysArr; @end @implementation ViewController

  • (void)viewDidLoad { [superviewDidLoad]; NSMutableArray*mArr = @[].mutableCopy; [mArr addObject:@1]; [mArr addObject:@2]; [mArr addObject:@3]; self.cArr = mArr;self.sArr = mArr; [mArr addObject:@100]; NSLog(@"self.cArr:%@, self.sArr:%@",self.cArr,self.sArr); //输出 self.cArr: (1,2,3) self.sArr:(1,2,3,100)}

由结果可看出,如果我们使用是 strong ,那么这个属性sArr就有可能指向一个可变对象mArr,如果这个可变对象在外部被修改了,就可能会在对象sArr不知情的情况下遭人更改。所以,这时就要拷贝一份“不可变” (immutable)的字符串,确保对象中的字符串值不会无意间变动。只要实现属性所用的对象是“可变的” (mutable),就应该在设置新属性值时拷贝一份。

4, 接着再上两个测试例子,对比看输出结果

  • (void)test01 {
    NSArray array= @[@1,@2,@3,@4];
    NSArray
    copyArr = [arraycopy];
    NSArray mCopyArr = [arraymutableCopy]; NSMutableArray mcArr = [arraycopy];
    NSMutableArray *mmCopyArr = [arraymutableCopy];
    NSLog(@"array:%p--copyArr:%p--mCopyArr:%p--mcArr:%p---mmCopyArr:%p",array,copyArr,mCopyArr,mcArr,mmCopyArr);

    / 输出结果array:0x608000058780--copyArr:0x608000058780--mCopyArr:0x608000058690--mcArr:0x608000058780---mmCopyArr:0x608000058570 /}

  • (void)test02 { NSArray tarray = @[@1,@2,@3,@4];
    NSMutableArray
    array= [[NSMutableArray alloc] init]; [arrayaddObjectsFromArray:tarray];
    NSArray copyArr = [arraycopy];
    NSArray
    mCopyArr = [arraymutableCopy];
    NSMutableArray mcArr = [arraycopy];
    NSMutableArray
    mmCopyArr = [arraymutableCopy]; NSLog(@"array:%p--copyArr:%p--mCopyArr:%p--mcArr:%p---mmCopyArr:%p",array,copyArr,mCopyArr,mcArr,mmCopyArr);

    / 输出结果array:0x600000054be0--copyArr:0x600000054b20--mCopyArr:0x600000054b50--mcArr:0x6000000554e0---mmCopyArr:0x600000055510 /}

总结:

NSArray的copy ---->指针拷贝---->浅拷贝,其余NSArray的mutableCopy,NSMutableArray的copy, NSMutableArray的mutableCopy 均为深拷贝,即内容拷贝。

关于NSString(非集合类对象),NSDictionary及其对应的可变类型都可以此类推。

相关阅读链接:

https://www.zybuluo.com/MicroCai/note/50592

收藏
0
sina weixin mail 回到顶部