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