被無數人寫過的assign,retain,strong,weak,unsafe_unretained,還有copy

斯人如是丶發表於2016-05-13

文/賣萌涼(簡書作者)
原文連結:http://www.jianshu.com/p/4a1d1921284b


雖然的確是最基本&被無數人寫過的問題,但是今天還是想弄得更清楚一些,所以看了看官方文件,寫了這篇部落格。

assign,retain,strong,weak,unsafe_unretained,還有copy,這些都是一個property在宣告中可以指定的屬性,且都與記憶體管理有關。下面會從Non-ARC和ARC兩種情況討論一下這些屬性的意義。


Non-ARC


官方文件的描述上看,Non-ARC的記憶體管理模式下,編譯器會為帶有不同屬性的property自動生成對應的accessor方法。並且蘋果十分建議在可能的情況下通過accessor方法來操縱property,而不是操縱它對應的例項變數。

如果需要對某些property自定義accessor方法,則需要程式設計師注意這個property的屬性。個人認為,寫在property旁邊的屬性,並不是真正控制著這個property的行為,它只是對編譯器自動生成的accessor方法提供了指導,當然也為自定義accessor方法的程式設計師和他的客戶程式設計師提供了指導。

  • assign

在Non-ARC記憶體管理模式下,assign是一個property的預設屬性,無論這個property代表一個簡單資料型別,還是一個指向物件的指標。也就是說:

@property (nonatomic) NSNumber *count;

等價於:

@property (nonatomic, assign) NSNumber *count;

assign主要應用於代表簡單資料型別的property,比如int,float等。
如果這個用assign屬性修飾的property代表一個指向物件的指標,那麼當這個指標指向某個物件時,這個物件的引用計數不應該被改變。也就是說,用assign屬性修飾的property,不應該持有一個物件。
因為這個property不持有物件,所以它所指向的物件很可能已經在別處被釋放了。這時它就有可能成為一枚懸垂指標,訪問它指向的記憶體地址時,可能會發生意想不到的狀況。

  • retain

retain不能修飾用來代表簡單資料型別的property,否則編譯器會報錯:

@property (nonatomic, retain) int num;//編譯器報錯:Property with 'retain (or strong)' attribute must be of object type

如果一個property被retain修飾,這代表著這個property應該持有它所指向的物件。
官方文件中展示了一個被retain修飾的property:

@property (nonatomic, retain) NSNumber *count;

編譯器可能為它實現的accessor方法:

- (NSNumber *)count {
    return _count;
}
- (void)setCount:(NSNumber *)newCount {
    [newCount retain];
    [_count release];
    // Make the new assignment.
    _count = newCount;
}

注意,考慮到newCount和_count可能指向同一個物件,所以在setter方法中,必須首先呼叫retain,以防這個物件被釋放。

  • copy

copy也不能修飾用來代表簡單資料型別的property,否則編譯器會報錯:

@property (nonatomic, copy) int num;//編譯器報錯:Property with 'copy' attribute must be of object type

如果一個property被copy修飾,那麼賦值到這個property的物件,應該是原有物件的一份拷貝。
只有實現了NSCopying協議,並且實現了其中的copyWithZone:方法的物件才能被拷貝。
但是並不是所有的拷貝都產生了新的物件,有些類在實現copyWithZone:方法時,有著它們自己的考慮。比如NSString

@property (nonatomic, copy) NSString *myString;
NSString *string = [[NSString alloc] initWithString:@"Hello"];
self.myString = string;
NSLog(@"%d", string == _myString);//輸出1

在這裡,property的指標和原先的指標指向的是同一個地址。

  • unsafe_unretained

個人認為unsafe_unretained與assign是等價的。

  • strong

個人認為strong與retain是等價的。
官方文件中有這樣的示例程式碼:

// The following declaration is a synonym for: @property(retain) MyClass *myObject;
@property(strong) MyClass *myObject;

表示了strong和retain是同義詞。

  • weak

Non-ARC記憶體管理模式下無法使用weak來修飾一個property,編譯器會報錯。


ARC


ARC有效時,物件型別的變數將有所有權修飾符來修飾。一共有以下四種所有權修飾符:

__strong 修飾符
__weak 修飾符
__unsafe_unretained 修飾符
__autoreleasing 修飾符

四種修飾符的具體意思,就不在這裡解釋了(´・_・`)

編譯器在為一個property合成例項變數時,也會使用所有權修飾符來修飾這個例項變數。根據property屬性的不同,用來修飾例項變數的所有權修飾符也不盡相同。

  • strong

在ARC記憶體管理模式下,strong是一個代表物件型別的property的預設屬性,並且它不能修飾用來代表簡單資料型別的property。編譯器在合成例項變數時,將使用__strong修飾符。
如果另外自定義了用其他修飾符修飾的例項變數,編譯器會報錯。可以用這個方法來驗證property的各個屬性對應的例項變數的所有權修飾符。

@interface ViewController ()
{
    __weak NSObject *_obj;//編譯器報錯:Existing instance variable '_obj' for strong property 'obj' may not be weak
}

@property (nonatomic, strong) NSObject *obj;

@end
  • weak

weak也不能修飾用來代表簡單資料型別的property。
編譯器將為weak修飾的property生成帶__weak所有權修飾符的例項變數。

  • copy

copy也不能修飾用來代表簡單資料型別的property。
編譯器將為copy修飾的property生成帶__strong所有權修飾符的例項變數。
編譯器自動合成的setter方法會呼叫物件的copyWithZone:方法。雖然第三方程式設計師可以自定義setter方法,但是為了程式的可讀性,也應該在其中執行拷貝的邏輯。

  • retain

和Non-ARC的理由一樣,個人認為retain和strong是等價的。

  • unsafe_unretained

編譯器將為unsafe_unretained修飾的property生成帶__unsafe_unretained所有權修飾符的例項變數。
與weak和strong不同的是,unsafe_unretained也可以修飾代表簡單資料型別的property。

  • assign

個人認為assign和unsafe_unretained等價。
assign在ARC記憶體管理模式下,仍然是代表簡單資料型別的property的預設屬性。


參考:

Transitioning to ARC Release Notes
Practical Memory Management
Objective-C Automatic Reference Counting (ARC)
Object copying
Objective-C高階程式設計 iOS與OS X多執行緒和記憶體管理



相關文章