ARC有效時,id類型和對象類型必須附加所有權(quán)修飾符,所有權(quán)修飾符一共4種:
-
__strong
修飾符.(強引用會持有對象) -
__weak
修飾符.(弱引用不會持有對象) -
__unsafe_unretained
修飾符 (不會持有對象荷腊,有懸掛指針的風(fēng)險) -
__autoreleasing
修飾符 (對象會被注冊到自動釋放池里)
其中__strong
修飾符是id類型和對象類型默認的所有權(quán)修飾符.
使用__strong,__weak,__autoreleasing
修飾符的自動變量,會被初始化為nil.
何為內(nèi)存泄露?
內(nèi)存泄露指的是應(yīng)當(dāng)廢棄的對象在超出其生存周期后繼續(xù)存在.
僅使用__strong
修飾符,會導(dǎo)致循環(huán)引用的發(fā)生.循環(huán)引用有兩種:
- 相互循環(huán)引用,即兩個對象間的相互強引用.
- 自引用導(dǎo)致的循環(huán)引用.
__weak
修飾符
適用范圍:只能修飾對象類型,不能用于基本數(shù)據(jù)類型.
__weak
修飾符優(yōu)點:
1.可以避免循環(huán)引用導(dǎo)致的內(nèi)存泄露.因為弱引用不會持有對象.
2.在持有某對象的弱引用時,若該對象被廢棄,則此弱引用將自動失效且指針變量將被賦值為nil.
__weak
修飾符只能用于iOS5及以上,在iOS4只能使用__unsafe_unretained
代替.不過現(xiàn)在已經(jīng)到了最低適配版本iOS7了,所以可以放心使用__weak
修飾符.
__weak
修飾符缺點:
1.正因為附有__weak
修飾符的指針變量,當(dāng)它指向的對象被銷毀時,系統(tǒng)會將該對象所有的__weak
指針都置為nil,所以效率相比__unsafe_unretained
修飾符要低.有時候優(yōu)點也是缺點.就看你如何掌握好"度".
2.不能用于基本數(shù)據(jù)類型.
__unsafe_unretained
修飾符
適用范圍:可用于基本數(shù)據(jù)類型也可用于對象類型.不過一般使用 __unsafe_unretained
修飾對象類型,如果是基本類型則Xcode會有警告弯予。
附有__unsafe_unretained
修飾符的變量也不能持有對象.
缺點:和__weak
修飾符不同的是如果帶__unsafe_unretained
修飾符的變量指向的對象被廢棄了那么該指針變量的值不會被置為nil,依然還是以前的值.但它已經(jīng)是野指針了,再次訪問將會崩潰,雖然不是每次都崩.
__autoreleasing
修飾符
ARC有效時,將對象賦值給附有__autoreleasing
修飾符的指針變量等價于在ARC無效時調(diào)用對象的autorelease方法,即將對象注冊到autoreleasepool.
但是,顯式地附加__autoreleasing
修飾符同顯式地__strong
修飾符一樣罕見.因為編譯器會幫我們添加.
比如使用alloc.../開頭以外的方法來取得的對象是已經(jīng)被注冊到了autoreleasepool里的(雖然ARC下該返回的對象不一定真的注冊到autoreleasepool里,這里暫且這么理解).這是由于編譯器會檢查方法名是否以alloc.../開始,如果不是則自動將作為返回值的對象注冊到autoreleasepool.
比如ARC有效時,下面的方法:
+ (id)array
{
id obj = [[NSMutableArray alloc] init];
return obj;
}
由于沒有顯式的指定所有權(quán)修飾符,所以 id obj
等同于id __strong obj
.由于return使得對象變量超出其作用域,所以該強引用指針變量指向的對象將被釋放,但該對象作為函數(shù)的返回值,編譯器會自動將其注冊到autoreleasepool.這里也都沒有使用顯式地附加__autoreleasing
修飾符.
另外一種可以不需要顯式的使用__autoreleasing
修飾符的情況就是:id的指針或?qū)ο蟮闹羔?也就是雙重指針)在沒有顯式指定時會被附加上__autoreleasing
修飾符.
最常見的例子就是,獲取錯誤NSError時.
NSError *err = nil; Bool result = [obj performOperationWithError:&err];
該方法的聲明為:
- (BOOL)performOperationWithError:(NSError **)error;
上述方法聲明是等同于
- (BOOL)performOperationWithError:(NSError * __autoreleasing *)error;
這里再說一次:作為alloc/new/copy/mutableCopy方法返回值取得的對象是自己生成并持有的,其他情況下便是取得非自己生成并持有的對象.因此,使用附有__autoreleasing
修飾符的變量作為對象取得參數(shù),與除alloc/new/copy/mutableCopy外其他方法的返回值取得對象完全一樣,都會注冊到autoreleasepool,并取得非自己生成并持有的對象.
最后,賦值給對象指針時,所有權(quán)修飾符必須一致.
因此下面的源代碼會產(chǎn)生編譯器錯誤:
NSError *err = nil;
NSError **p = &err;
需要改為:
NSError *err = nil;
NSError * __strong *p = &err; //這里并不會導(dǎo)致對象的引用計數(shù)+1
//比如
NSObject * __strong *p1 = NULL;
{
NSObject *obj = MATBaseViewController.new;
p1 = &obj; //并不會導(dǎo)致obj引用計數(shù)+1
}
NSLog(@"obj=%@", *p1); //obj=(null)
然而下面的這種情況又是怎么回事?
NSError *err = nil;默認是 __strong修飾符.而方法的參數(shù)聲明是__autoreleasing修飾符.
Bool result = [obj performOperationWithError:&err];
該方法的聲明為:
- (BOOL)performOperationWithError:(NSError **)error;
實際上,是編譯器自動將上述源代碼做了轉(zhuǎn)換變成:
NSError __strong *error = nil;
NSError __autoreleasing *tmp = error;
BOOL result = [obj performOperationWithError:&tmp]; //這里tmp的所有權(quán)修飾符就和參數(shù)的一致了
error = tmp;
屬性關(guān)鍵字:
屬性關(guān)鍵字有:
1.assign 對應(yīng)__unsafe_unretained
修飾符洛勉,和unsafe_unretained幾乎沒有區(qū)別,可以用于對象類型也可以用于基本類型的屬性聲名然磷。
由于assign不是所有權(quán)修飾符所以沒有 __assign NSObject *_obj;
這樣的寫法。正確的寫法為__unsafe_unretained NSObject *_obj;
。
2.copy 和MRC里的一樣落午,實際會調(diào)用對象的copy方法,所以要求對象需要實現(xiàn)NSCoping協(xié)議肚豺。
如果有重寫copy屬性的setter方法溃斋,則在賦值時需要調(diào)用copy方法,而不是簡單的賦值吸申。
- (void)setAcat:(ARCCat *)acat {
_acat = [acat copy];
}
3.retain 對應(yīng)__strong
修飾符 在ARC模式下,依然可以使用.
4.strong 默認 對應(yīng)__strong
修飾符
5.unsafe_unretained 對應(yīng)__unsafe_unretained
修飾符
6.weak 對應(yīng)__weak
修飾符
7.atomic 默認
8.nonatomic
9.readonly
10.readwrite 默認
需要注意的是屬性的關(guān)鍵字需要和它的實例變量的修飾符一致(當(dāng)你不使用系統(tǒng)幫你生成的實例變量時)
@interface ViewController ()
{
__autoreleasing NSString *_ttrsr;
}
@property (nonatomic, strong) NSString *ttrsr;
@end
這樣寫會報錯.
需要改為__strong NSString *_ttrsr;
或NSString *_ttrsr;