雖然的確是最基本&被無(wú)數(shù)人寫(xiě)過(guò)的問(wèn)題,但是今天還是想弄得更清楚一些,所以看了看官方文檔,寫(xiě)了這篇博客。
assign,retain,strong,weak,unsafe_unretained,還有copy,這些都是一個(gè)property在聲明中可以指定的屬性,且都與內(nèi)存管理有關(guān)。下面會(huì)從Non-ARC和ARC兩種情況討論一下這些屬性的意義。
Non-ARC
從官方文檔的描述上看,Non-ARC的內(nèi)存管理模式下,編譯器會(huì)為帶有不同屬性的property自動(dòng)生成對(duì)應(yīng)的accessor方法。并且蘋(píng)果十分建議在可能的情況下通過(guò)accessor方法來(lái)操縱property,而不是操縱它對(duì)應(yīng)的實(shí)例變量。
如果需要對(duì)某些property自定義accessor方法,則需要程序員注意這個(gè)property的屬性。個(gè)人認(rèn)為,寫(xiě)在property旁邊的屬性,并不是真正控制著這個(gè)property的行為,它只是對(duì)編譯器自動(dòng)生成的accessor方法提供了指導(dǎo),當(dāng)然也為自定義accessor方法的程序員和他的客戶程序員提供了指導(dǎo)。
- assign
在Non-ARC內(nèi)存管理模式下,assign是一個(gè)property的默認(rèn)屬性,無(wú)論這個(gè)property代表一個(gè)簡(jiǎn)單數(shù)據(jù)類型,還是一個(gè)指向?qū)ο蟮闹羔槨R簿褪钦f(shuō):
@property (nonatomic) NSNumber *count;
等價(jià)于:
@property (nonatomic, assign) NSNumber *count;
assign主要應(yīng)用于代表簡(jiǎn)單數(shù)據(jù)類型的property,比如int,float等。
如果這個(gè)用assign屬性修飾的property代表一個(gè)指向?qū)ο蟮闹羔槪敲串?dāng)這個(gè)指針指向某個(gè)對(duì)象時(shí),這個(gè)對(duì)象的引用計(jì)數(shù)不應(yīng)該被改變。也就是說(shuō),用assign屬性修飾的property,不應(yīng)該持有一個(gè)對(duì)象。
因?yàn)檫@個(gè)property不持有對(duì)象,所以它所指向的對(duì)象很可能已經(jīng)在別處被釋放了。這時(shí)它就有可能成為一枚懸垂指針,訪問(wèn)它指向的內(nèi)存地址時(shí),可能會(huì)發(fā)生意想不到的狀況。
- retain
retain不能修飾用來(lái)代表簡(jiǎn)單數(shù)據(jù)類型的property,否則編譯器會(huì)報(bào)錯(cuò):
@property (nonatomic, retain) int num;//編譯器報(bào)錯(cuò):Property with 'retain (or strong)' attribute must be of object type
如果一個(gè)property被retain修飾,這代表著這個(gè)property應(yīng)該持有它所指向的對(duì)象。
官方文檔中展示了一個(gè)被retain修飾的property:
@property (nonatomic, retain) NSNumber *count;
編譯器可能為它實(shí)現(xiàn)的accessor方法:
- (NSNumber *)count {
return _count;
}
- (void)setCount:(NSNumber *)newCount {
[newCount retain];
[_count release];
// Make the new assignment.
_count = newCount;
}
注意,考慮到newCount和_count可能指向同一個(gè)對(duì)象,所以在setter方法中,必須首先調(diào)用retain,以防這個(gè)對(duì)象被釋放。
- copy
copy也不能修飾用來(lái)代表簡(jiǎn)單數(shù)據(jù)類型的property,否則編譯器會(huì)報(bào)錯(cuò):
@property (nonatomic, copy) int num;//編譯器報(bào)錯(cuò):Property with 'copy' attribute must be of object type
如果一個(gè)property被copy修飾,那么賦值到這個(gè)property的對(duì)象,應(yīng)該是原有對(duì)象的一份拷貝。
只有實(shí)現(xiàn)了NSCopying協(xié)議,并且實(shí)現(xiàn)了其中的copyWithZone:
方法的對(duì)象才能被拷貝。
但是并不是所有的拷貝都產(chǎn)生了新的對(duì)象,有些類在實(shí)現(xiàn)copyWithZone:
方法時(shí),有著它們自己的考慮。比如NSString
:
@property (nonatomic, copy) NSString *myString;
NSString *string = [[NSString alloc] initWithString:@"Hello"];
self.myString = string;
NSLog(@"%d", string == _myString);//輸出1
在這里,property的指針和原先的指針指向的是同一個(gè)地址。
- unsafe_unretained
個(gè)人認(rèn)為unsafe_unretained與assign是等價(jià)的。
- strong
個(gè)人認(rèn)為strong與retain是等價(jià)的。
官方文檔中有這樣的示例代碼:
// The following declaration is a synonym for: @property(retain) MyClass *myObject;
@property(strong) MyClass *myObject;
表示了strong和retain是同義詞。
- weak
Non-ARC內(nèi)存管理模式下無(wú)法使用weak來(lái)修飾一個(gè)property,編譯器會(huì)報(bào)錯(cuò)。
ARC
ARC有效時(shí),對(duì)象類型的變量將有所有權(quán)修飾符來(lái)修飾。一共有以下四種所有權(quán)修飾符:
__strong 修飾符
__weak 修飾符
__unsafe_unretained 修飾符
__autoreleasing 修飾符
四種修飾符的具體意思,就不在這里解釋了(′?_?`)
編譯器在為一個(gè)property合成實(shí)例變量時(shí),也會(huì)使用所有權(quán)修飾符來(lái)修飾這個(gè)實(shí)例變量。根據(jù)property屬性的不同,用來(lái)修飾實(shí)例變量的所有權(quán)修飾符也不盡相同。
- strong
在ARC內(nèi)存管理模式下,strong是一個(gè)代表對(duì)象類型的property的默認(rèn)屬性,并且它不能修飾用來(lái)代表簡(jiǎn)單數(shù)據(jù)類型的property。編譯器在合成實(shí)例變量時(shí),將使用__strong
修飾符。
如果另外自定義了用其他修飾符修飾的實(shí)例變量,編譯器會(huì)報(bào)錯(cuò)。可以用這個(gè)方法來(lái)驗(yàn)證property的各個(gè)屬性對(duì)應(yīng)的實(shí)例變量的所有權(quán)修飾符。
@interface ViewController ()
{
__weak NSObject *_obj;//編譯器報(bào)錯(cuò):Existing instance variable '_obj' for strong property 'obj' may not be weak
}
@property (nonatomic, strong) NSObject *obj;
@end
- weak
weak也不能修飾用來(lái)代表簡(jiǎn)單數(shù)據(jù)類型的property。
編譯器將為weak修飾的property生成帶__weak
所有權(quán)修飾符的實(shí)例變量。
- copy
copy也不能修飾用來(lái)代表簡(jiǎn)單數(shù)據(jù)類型的property。
編譯器將為copy修飾的property生成帶__strong
所有權(quán)修飾符的實(shí)例變量。
編譯器自動(dòng)合成的setter方法會(huì)調(diào)用對(duì)象的copyWithZone:
方法。雖然第三方程序員可以自定義setter方法,但是為了程序的可讀性,也應(yīng)該在其中執(zhí)行拷貝的邏輯。
- retain
和Non-ARC的理由一樣,個(gè)人認(rèn)為retain和strong是等價(jià)的。
- unsafe_unretained
編譯器將為unsafe_unretained修飾的property生成帶__unsafe_unretained
所有權(quán)修飾符的實(shí)例變量。
與weak和strong不同的是,unsafe_unretained也可以修飾代表簡(jiǎn)單數(shù)據(jù)類型的property。
- assign
個(gè)人認(rèn)為assign和unsafe_unretained等價(jià)。
assign在ARC內(nèi)存管理模式下,仍然是代表簡(jiǎn)單數(shù)據(jù)類型的property的默認(rèn)屬性。
參考:
Transitioning to ARC Release Notes
Practical Memory Management
Objective-C Automatic Reference Counting (ARC)
Object copying
Objective-C高級(jí)編程 iOS與OS X多線程和內(nèi)存管理