1.引
看一段代碼
@interface ViewController ()
@property (nonatomic, copy) NSMutableArray *array;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
self.array = [NSMutableArray array];
[self.array addObject:@"A"];
@end
結(jié)果是系統(tǒng)崩潰,后臺打印如下錯(cuò)誤信息
2018-04-03 15:02:11.238078+0800 iOS[55183:3743941] -[__NSArray0 addObject:]: unrecognized selector sent to instance 0x6040000065a0
2018-04-03 15:02:11.242691+0800 iOS[55183:3743941] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[__NSArray0 addObject:]: unrecognized selector sent to instance 0x6040000065a0'
這個(gè)時(shí)候如果將array的初始化方法換成
_array = [NSMutableArray array];
運(yùn)行正常!
2.原因
不討論copy和strong的區(qū)別,百度可以找到一堆,這里只討論為什么_array不會崩潰!那貌似標(biāo)題就得改了。。。
在Java里有一個(gè)名詞叫Bean,對屬性生成setter和getter方法之后,這個(gè)變量就是一個(gè)bean;
回到OC,系統(tǒng)會自動對@property變量生成setter和getter方法,也就是一個(gè)bean,它們其實(shí)都是在這個(gè)變量的基礎(chǔ)上封了一層,我們用到懶加載的時(shí)候是要對自己的變量自己寫讀和寫的方法
//getter:
- (NSMutableArray *)array {
NSLog("%@", _array);
return _array;
}
//setter:
- (void)setArray:(NSMutableArray *)newValue {
_array = newValue;
}
用到的就是_array.所以直接使用_array是繞過了getter方法和setter方法,那么_array是直接操作了實(shí)例變量;
再來,系統(tǒng)申明一個(gè)變量的時(shí)候默認(rèn)是__strong也就是說_array是一個(gè)被strong修飾的變量,并不是copy修飾的。
id objc = [NSObject new];
//完整的代碼是這個(gè)
id __strong objc = [NSObject new];
理解了上面的原因之后,總結(jié)一下,self.array是使用了系統(tǒng)的setter和getter,所以array初始化的時(shí)候,將[NSMutableArray array]這個(gè)對象copy了一下,變成了[array copy]傳給了_array,NSArray是NSMutableArray的父類,沒有addObject的方法,所以崩潰了~
3. 解決
最好的方法是,將修飾符
@property (nonatomic, strong) NSMutableArray *array;