前言
property
屬性關(guān)鍵字我們在日常的開發(fā)中經(jīng)常會用到,所以我們有必要對其有充分的了解,這樣對于我們?nèi)粘i_發(fā)使用時就能做到知其所以然。
property關(guān)鍵字介紹
property
關(guān)鍵字分為四類:
- 原子性:
atomic
和nonatomic
,property
中默認(rèn)是atomic
,也就是線程安全,但是我們一般使用的是nonatomic
。因?yàn)?code>atomic的線程安全系統(tǒng)資源開銷相對較大,影響性能,即使我們需要使用線程安全,我們也可以使用其他方法實(shí)現(xiàn)。atomic實(shí)現(xiàn)原理是在getter
、setter
中使用@synchronized
同步鎖關(guān)鍵字進(jìn)行代碼塊鎖定而實(shí)現(xiàn)的。
補(bǔ)充atomic是自旋鎖,即當(dāng)上一線程沒有執(zhí)行完畢(被鎖住),下一線程會一直等待(不會進(jìn)入睡眠狀態(tài)),當(dāng)上一線程執(zhí)行完畢,下一線程立即執(zhí)行。他區(qū)別于互斥鎖,互斥鎖在等待的時候,會進(jìn)入睡眠狀態(tài),當(dāng)上一個線程執(zhí)行完畢,睡眠狀態(tài)就會被喚醒,然后再執(zhí)行。
引用計(jì)數(shù)相關(guān):
assign
、retain
、weak
、strong
、copy
、iOS5以前使用的unsafe_unretained
。
assign:主要修飾基本數(shù)據(jù)類型,不能修飾對象類型,如NSInterger
,CGFloat
等類型。并且統(tǒng)一由系統(tǒng)棧進(jìn)行內(nèi)存管理。
retain:修飾對象類型,強(qiáng)引用對象,并是對象引用計(jì)數(shù)加1,可用于MRC
環(huán)境中。
weak:修飾對象類型,對對象弱引用,不增加對象的引用計(jì)數(shù)。如果對象銷毀了,指針會自動指向nil
,所以可以防止野指針的問題。
strong:修飾對象類型,對對象強(qiáng)引用,會增加對象的引用計(jì)數(shù)。如果指向了空對象,會造成野指針。只能用于ARC
環(huán)境。
copy:在一個新對象引用計(jì)數(shù)為1,賦值時對傳入值進(jìn)行一份拷貝,所以才使用copy關(guān)鍵字。你將一個對象賦值給一個屬性,該屬性并不會持有對象,而是會創(chuàng)建一個新對象,并將這個對象拷貝給它。使用copy
關(guān)鍵字的對象必須實(shí)現(xiàn)NSCoding
協(xié)議。
unsafe_unretained:跟weak
類似,聲明一個弱引用,區(qū)別是當(dāng)引用計(jì)數(shù)為0時,變量不會自動設(shè)置為nil
。讀寫權(quán)限相關(guān):默認(rèn)是
readwrite
(可讀可寫),還有readonly,修飾屬性時,屬性不能被外界修改。方法名:可設(shè)置屬性的
setter
和getter
方法名。
補(bǔ)充介紹
weak
關(guān)鍵字:
- 使用場景
用于一些對象互相引用時,避免出現(xiàn)互相強(qiáng)引用而導(dǎo)致的循環(huán)引用,對象不能釋放的。 - 實(shí)現(xiàn)原理
weak
修飾時,runtime
會維護(hù)一個hash
表(也稱為weak
表),用于存儲對象的所有weak
指針,hash
表的key
是該對象的地址,value
為weak
指針的地址(這個地址的值是所指對象的地址)數(shù)組。(備注strong
是通過runtime
維護(hù)的一個自動引用計(jì)數(shù)表)
weak
的實(shí)現(xiàn)原理總結(jié):
- 初始化時,
runtime
會調(diào)用objc_initWeak
函數(shù),初始化一個新的weak
指針指向?qū)ο蟮刂罚?/li> - 添加引用時,
objc_initWeak
函數(shù)會調(diào)用objc_storeWeak
函數(shù),objc_storeWeak
的作用是更新指針指向,創(chuàng)建對應(yīng)的弱引用表(hash表); - 釋放時,調(diào)用
clearDeallocating
函數(shù)。clearDeallocating
函數(shù)首先根據(jù)對象地址獲取weak
指針地址的數(shù)組,然后遍歷這個數(shù)組把其中指向空對象的指針設(shè)為nil
,最后把這個指針從weak
表中刪除,最后清理對象的記錄。
copy
和strong
:
講這兩個字之前我們需要了解深復(fù)制和淺復(fù)制相關(guān)的只是,可以參考這里。具體示例如下:
@property (nonatomic, copy) NSMutableArray *mutArray;
NSMutableArray *mutArray1 = [NSMutableArray array];
self.mutArray = mutArray1;
等同于
@property (nonatomic, strong) NSMutableArray *mutArray;
NSMutableArray *mutArray1 = [NSMutableArray array];
self.mutArray = [mutArray1 copy];
經(jīng)過測試之后我們知道使用copy修飾的mutArray
數(shù)組,當(dāng)調(diào)用它的setter
方法,它會建立一個引用計(jì)數(shù)為1的新對象,然后釋放舊對象。而copy
修飾的屬性賦值時經(jīng)過copy
其實(shí)已經(jīng)變成了不可變數(shù)組。而使用可變數(shù)組的增、刪、改、查函數(shù)是會發(fā)現(xiàn)找不到相關(guān)的實(shí)例方法而crash
。
NSString 為什么用 copy 而不用 retain
我們通過實(shí)例來看看:
@property (nonatomic, retain) NSString *string;
NSMutableString *string1 = [[NSMutableString alloc] initWithString:@"abc"];
self.string = string1;
[string1 appendString:@"123"];
NSLog(@"============== %@ =========", self.string);
2019-06-19 17:08:58.114333+0800 ThinTableVIew1[96955:2656816] ============== abc123 =========
從打印的信息可以看到當(dāng)改變string1
的值時,self.string
的值也改變了。下面我們通過查看string
屬性setter
方法的實(shí)現(xiàn)來探究一下原理:
@property (nonatomic, retain) NSString *string;
- (void)setString:(NSString *)string {
if (_string != string) {
[_string release];
_string = [string retain];
//相當(dāng)于
//[string retain];
//_string = string;
}
}
==================
@property (nonatomic, copy) NSString *string;
- (void)setString:(NSString *)string {
if (_string != string) {
[_string release];
_string = [string copy];
}
}
我們可以知道:
- 當(dāng)使用
retain
修飾string
時調(diào)用的是_string = [string retain]
,這樣只會增加string的引用,而_string
指針和string
是指向同一塊內(nèi)容。所以改變string
的內(nèi)容同樣的會改變_string
的內(nèi)容。 - 當(dāng)使用
copy
修飾string
時,當(dāng)傳入的對象是可變對象時,調(diào)用的是[string copy]
;會創(chuàng)建一個新的對象賦值給_string
,所以_string
和string
不會互相干擾,而改變string
的內(nèi)容不會影響_string
。