Property中好玩的修飾符-copy和strong

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;

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容