被無數人寫過的assign,retain,strong,weak,unsafe_unretained,還有copy
文/賣萌涼(簡書作者)
原文連結: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多執行緒和記憶體管理
相關文章
- iOS5 strong, weak, unsafe_unretained ARCiOSAI
- IOS基礎:retain,copy,assign及autoreleaseiOSAI
- Swift中的Weak/Strong DanceSwift
- 對 Strong-Weak Dance的思考
- 對 Strong-Weak Dance 的思考
- retain 和copy的區別AI
- iOS strong weak unowned引用iOS
- IOS @proporty 關鍵字(一)retain strongiOSAI
- weak和assign的區別
- 深入理解 weak-strong dance
- assign 和 weak 區別?
- iOS strong和copy的區別iOS
- 宣告 NSString 型別的屬性,到底用 strong 還是 copy ?型別
- 【iOS】用strong和weak來修飾成員變數的對比iOS變數
- iOS中copy和strong的個人理解iOS
- Why should we copy blocks rather than retain?BloCAI
- iOS之property裡的copy、strong區別iOS
- iOS中assign和weak修飾符的區別iOS
- 深拷貝和淺拷貝 copy與strong修飾變數的區別變數
- 面試題分解—「淺複製/深複製、定義屬性使用copy還是strong ?」面試題
- 最新的assign與weak的區別 看不明白你打我
- 無人駕駛技術被提上議程: 或改寫無人駕駛領域歷史
- Freemarker的迴圈通過assign指令引入計數變數變數
- [論文閱讀] Aligner@ Achieving Efficient Alignment through Weak-to-Strong Correction
- 無符號數相減得到的是無符號還是有符號?符號
- iOS利用runtime給分類擴充套件屬性Strong bool copyiOS套件
- __unsafe_unretainedAI
- iOS main()執行前的過程 + weak 置 nil的過程iOSAI
- block中self關鍵字的使用-防止self 被retain一次BloCAI
- 你的工作還有多久會被機器取代?
- 3款超好用的Chrome擴充套件,有多少人還沒用過?Chrome套件
- Effective C++是copy constructor 還是 copy assignment(“=”)的判斷C++Struct
- 全面推廣自動駕駛還不成熟:無人駕駛還要跨過三道坎自動駕駛
- 硬紀元AI峰會前瞻:火爆的無人駕駛還將有哪些深度創新?AI
- Rust 程式設計影片教程(進階)——015_3 檢視 strong_count 和 weak_count 的改變Rust程式設計
- 被世界遺忘的掃雷遊戲,只有中國人還在沉迷遊戲
- 模型被投毒攻擊,如今有了新的安全手段,還被AI頂刊接收模型AI
- 一個被寫爛的redux計數小例子Redux