property
屬性關鍵字咱們在平常的開發中常常會用到,因此咱們有必要對其有充分的瞭解,這樣對於咱們平常開發使用時就能作到知其因此然。數組
property
關鍵字分爲四類:安全
atomic
和nonatomic
,property
中默認是atomic
,也就是線程安全,可是咱們通常使用的是nonatomic
。由於atomic
的線程安全系統資源開銷相對較大,影響性能,即便咱們須要使用線程安全,咱們也可使用其餘方法實現。atomic實現原理是在getter
、setter
中使用@synchronized
同步鎖關鍵字進行代碼塊鎖定而實現的。補充atomic是自旋鎖,即當上一線程沒有執行完畢(被鎖住),下一線程會一直等待(不會進入睡眠狀態),當上一線程執行完畢,下一線程當即執行。他區別於互斥鎖,互斥鎖在等待的時候,會進入睡眠狀態,當上一個線程執行完畢,睡眠狀態就會被喚醒,而後再執行。bash
引用計數相關: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
。app
讀寫權限相關:默認是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;
複製代碼
等同於ui
@property (nonatomic, strong) NSMutableArray *mutArray;
NSMutableArray *mutArray1 = [NSMutableArray array];
self.mutArray = [mutArray1 copy];
複製代碼
通過測試以後咱們知道使用copy修飾的mutArray
數組,當調用它的setter
方法,它會創建一個引用計數爲1的新對象,而後釋放舊對象。而copy
修飾的屬性賦值時通過copy
其實已經變成了不可變數組。而使用可變數組的增、刪、改、查函數是會發現找不到相關的實例方法而crash
。atom
NSString 爲何用 copy 而不用 retain 咱們經過實例來看看:spa
@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
。