iOS property 關鍵字詳解

前言

property屬性關鍵字咱們在平常的開發中常常會用到,因此咱們有必要對其有充分的瞭解,這樣對於咱們平常開發使用時就能作到知其因此然。數組

property關鍵字介紹

property關鍵字分爲四類:安全

  1. 原子性:atomicnonatomicproperty中默認是atomic,也就是線程安全,可是咱們通常使用的是nonatomic。由於atomic的線程安全系統資源開銷相對較大,影響性能,即便咱們須要使用線程安全,咱們也可使用其餘方法實現。atomic實現原理是在gettersetter中使用@synchronized同步鎖關鍵字進行代碼塊鎖定而實現的。

補充atomic是自旋鎖,即當上一線程沒有執行完畢(被鎖住),下一線程會一直等待(不會進入睡眠狀態),當上一線程執行完畢,下一線程當即執行。他區別於互斥鎖,互斥鎖在等待的時候,會進入睡眠狀態,當上一個線程執行完畢,睡眠狀態就會被喚醒,而後再執行。bash

  1. 引用計數相關:assignretainweakstrongcopy、iOS5之前使用的unsafe_unretainedassign:主要修飾基本數據類型,不能修飾對象類型,如NSInterger,CGFloat等類型。而且統一由系統棧進行內存管理。 retain:修飾對象類型,強引用對象,並是對象引用計數加1,可用於MRC環境中。 weak:修飾對象類型,對對象弱引用,不增長對象的引用計數。若是對象銷燬了,指針會自動指向nil,因此能夠防止野指針的問題。 strong:修飾對象類型,對對象強引用,會增長對象的引用計數。若是指向了空對象,會形成野指針。只能用於ARC環境。 copy:在一個新對象引用計數爲1,賦值時對傳入值進行一份拷貝,因此才使用copy關鍵字。你將一個對象賦值給一個屬性,該屬性並不會持有對象,而是會建立一個新對象,並將這個對象拷貝給它。使用copy關鍵字的對象必須實現NSCoding協議。 unsafe_unretained:跟weak相似,聲明一個弱引用,區別是當引用計數爲0時,變量不會自動設置爲nilapp

  2. 讀寫權限相關:默認是readwrite(可讀可寫),還有readonly,修飾屬性時,屬性不能被外界修改。函數

  3. 方法名:可設置屬性的settergetter方法名。 ####補充介紹 weak關鍵字性能

  • 使用場景 用於一些對象互相引用時,避免出現互相強引用而致使的循環引用,對象不能釋放的。
  • 實現原理 weak修飾時,runtime會維護一個hash表(也稱爲weak表),用於存儲對象的全部weak指針,hash表的key是該對象的地址,valueweak指針的地址(這個地址的值是所指對象的地址)數組。(備註strong是經過runtime維護的一個自動引用計數表) weak的實現原理總結:
  1. 初始化時,runtime會調用objc_initWeak函數,初始化一個新的weak指針指向對象地址;
  2. 添加引用時,objc_initWeak函數會調用objc_storeWeak函數,objc_storeWeak的做用是更新指針指向,建立對應的弱引用表(hash表);
  3. 釋放時,調用clearDeallocating函數。clearDeallocating函數首先根據對象地址獲取weak指針地址的數組,而後遍歷這個數組把其中指向空對象的指針設爲nil,最後把這個指針從weak表中刪除,最後清理對象的記錄。

copystrong: 講這兩個字以前咱們須要瞭解深複製和淺複製相關的只是,能夠參考這裏。具體示例以下:測試

@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其實已經變成了不可變數組。而使用可變數組的增、刪、改、查函數是會發現找不到相關的實例方法而crashatom

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,因此_stringstring不會互相干擾,而改變string的內容不會影響_string
相關文章
相關標籤/搜索