美文网首页安小昕的小窝
为什么NSMutableArray用copy修饰会crash?

为什么NSMutableArray用copy修饰会crash?

作者: eileen01 | 来源:发表于2016-12-05 17:49 被阅读1820次

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

    废话不多说,先上代码

    @interface ViewController ()
    @property (nonatomic, copy) NSMutableArray *arr1;
    @end
    
    @implementation ViewController
    
    - (void)viewDidLoad {
        [super viewDidLoad];
        NSMutableArray *array = [NSMutableArray arrayWithObjects:@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) {
            [_arr1 release];
            _arr1 = [arr1 copy]; //内容拷贝,深拷贝
        }
    }
    
    
    • 如果修改成用strong修饰符,对应的setter方法实现如下
    - (void)setArr1:(NSMutableArray *)arr1 {
        if (_arr1 != arr1) {
            [_arr1 release];
            _arr1 = [arr1 retain]; //指针拷贝,浅拷贝
        }
    }
    

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

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

    • 通过copy方法可以创建可变对象或不可变对象的不可变副本, 对于不可变副本,其对象的值不可以改变。所以如果对copy返回值去调用可变对象的接口就会crash. (好像解答了上述问题)
    • mutableCopy 创建出来的是可变对象或者不可变对象的可变副本,对于可变副本,其对象是可变的。

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

    @interface ViewController ()
    @property (nonatomic, copy) NSArray *cArr;
    @property (nonatomic, strong) NSArray *sArr;
    @end
    
    @implementation ViewController
    
    - (void)viewDidLoad {
        [super viewDidLoad];
        
        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 = [array copy];
        NSArray *mCopyArr = [array mutableCopy];
        
        NSMutableArray *mcArr = [array copy];
        NSMutableArray *mmCopyArr = [array mutableCopy];
        
        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];
        [array addObjectsFromArray:tarray];
        
        NSArray *copyArr = [array copy];
        NSArray *mCopyArr = [array mutableCopy];
        
        NSMutableArray *mcArr = [array copy];
        NSMutableArray *mmCopyArr = [array mutableCopy];
        
        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

    相关文章

      网友评论

        本文标题:为什么NSMutableArray用copy修饰会crash?

        本文链接:https://www.haomeiwen.com/subject/qkxlmttx.html