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