OC所有權(quán)修飾符與屬性關(guān)鍵字

ARC有效時,id類型和對象類型必須附加所有權(quán)修飾符,所有權(quán)修飾符一共4種:

  1. __strong修飾符.(強引用會持有對象)
  2. __weak修飾符.(弱引用不會持有對象)
  3. __unsafe_unretained修飾符 (不會持有對象荷腊,有懸掛指針的風(fēng)險)
  4. __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)引用有兩種:

  1. 相互循環(huán)引用,即兩個對象間的相互強引用.
  2. 自引用導(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;

參考
Transitioning to ARC Release Notes

Objective-C Runtime Programming Guide--Declared Properties

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末梗劫,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子截碴,更是在濱河造成了極大的恐慌梳侨,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,826評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件日丹,死亡現(xiàn)場離奇詭異走哺,居然都是意外死亡,警方通過查閱死者的電腦和手機哲虾,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,968評論 3 395
  • 文/潘曉璐 我一進店門丙躏,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人束凑,你說我怎么就攤上這事晒旅。” “怎么了汪诉?”我有些...
    開封第一講書人閱讀 164,234評論 0 354
  • 文/不壞的土叔 我叫張陵废恋,是天一觀的道長。 經(jīng)常有香客問我扒寄,道長鱼鼓,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,562評論 1 293
  • 正文 為了忘掉前任该编,我火速辦了婚禮蚓哩,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘上渴。我一直安慰自己,他們只是感情好喜颁,可當(dāng)我...
    茶點故事閱讀 67,611評論 6 392
  • 文/花漫 我一把揭開白布稠氮。 她就那樣靜靜地躺著,像睡著了一般半开。 火紅的嫁衣襯著肌膚如雪隔披。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,482評論 1 302
  • 那天寂拆,我揣著相機與錄音奢米,去河邊找鬼抓韩。 笑死,一個胖子當(dāng)著我的面吹牛鬓长,可吹牛的內(nèi)容都是我干的谒拴。 我是一名探鬼主播,決...
    沈念sama閱讀 40,271評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼涉波,長吁一口氣:“原來是場噩夢啊……” “哼英上!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起啤覆,我...
    開封第一講書人閱讀 39,166評論 0 276
  • 序言:老撾萬榮一對情侶失蹤苍日,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后窗声,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體相恃,經(jīng)...
    沈念sama閱讀 45,608評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,814評論 3 336
  • 正文 我和宋清朗相戀三年笨觅,在試婚紗的時候發(fā)現(xiàn)自己被綠了拦耐。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,926評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡屋摇,死狀恐怖揩魂,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情炮温,我是刑警寧澤火脉,帶...
    沈念sama閱讀 35,644評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站柒啤,受9級特大地震影響倦挂,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜担巩,卻給世界環(huán)境...
    茶點故事閱讀 41,249評論 3 329
  • 文/蒙蒙 一方援、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧涛癌,春花似錦犯戏、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,866評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至弃衍,卻和暖如春呀非,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,991評論 1 269
  • 我被黑心中介騙來泰國打工岸裙, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留猖败,地道東北人。 一個月前我還...
    沈念sama閱讀 48,063評論 3 370
  • 正文 我出身青樓降允,卻偏偏與公主長得像恩闻,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子拟糕,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,871評論 2 354

推薦閱讀更多精彩內(nèi)容