为什么我把NSMutableArray的strong改成copy就崩溃

 -[__NSArray0 removeAllObjects]: unrecognized selector sent to instance 0x14e14970

网上查了下,大致还是内存管理的问题,看了还是一知半解。谁能解释一下全过程?

阅读 12.8k
8 个回答

先要搞明白copy属性到底做了些啥
一个属性标记了copy,当你调用他的setter方法,他会建立一个索引计数为1的对象,然后释放旧对象。
例如

@property (copy) NSMutableArray * a;

NSMutableArray* b = [NSMutableArray array];
a = b;

等同于

@property (strong) NSMutableArray * a;

NSMutableArray* b = [NSMutableArray array];
a = [b copy];

所以你的a根本就是一个NSArray,所以报了

-[__NSArray0 removeAllObjects]: unrecognized selector sent to instance 0x14e14970

未找到实例方法的错误。因为NSArray没有removeAllObjects方法。
正确的写法应该是不用copy属性

@property (strong) NSMutableArray * a;

NSMutableArray* b = [NSMutableArray array];
a =  [b mutableCopy];

这里的b可以是NSMutableArray也可以是NSArray

希望对你有帮助。

因为你copy之后, mutableArray 变成了Array.不信你打印你的array类型看看

通过copy方法可以创建可变对象或不可变对象的不可变副本,对于不可变副本,其对象的值不可以改变。

通过mutableCopy方法可以创建可变对象或不可变对象的可变副本,对于可变副本其对象是可变的。 

复制分为浅复制和深复制两种:浅复制只是复制对象的引用,并没有复制对象的具体内容。深复制则创建了要复制对象的具体内容,并返回对象副本的引用。

对于复制Foundation中的对象,默认并不是深复制,例如copy NSMutableArray对象是浅复制,只是对其引用进行复制;而copy NSMutableString对象是深复制,对原来引用的对象的具体内容也进行了复制。

想用copy属性,此时,可以重写setter方法

@property (nonatomic, copy) NSMutableArray *mArray;等同于

- (void)setMArray:(NSMutableArray *)mArray {
    _mArray = mArray.copy;
}

你的mArray属性就成了NSArray,NSMutableArray只能用strong修饰,不存在有copy修饰的情况,写了就成NSArray了。

如果是strong,直接是赋值_mArray = mArray;右边是什么,左边就是什么,并且是强引用新值

我猜你可能是self.mitablearray这种方式调用吧?这样其实是使用了getter方法,由于你的property设置为copy,会返回一个NSArray类型的变量,是不可变,所以调用remove方法时会出错,因为这个方法是NSMutableArray的方法。
前面的回复说的很详细了。

详情请阅读文章http://www.henishuo.com/interview-part-one/,第三题。

因为使用的是copy,所得到的实际是NSArray类型,它是不可变的,若在使用中使用了增、删、改操作,则会crash

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题
宣传栏