assign vs weak, __block vs __weak

assign vs weak

assign適用于基本數(shù)據(jù)類型鳖藕,weak是適用于NSObject對象,并且是一個弱引用。

assign其實也可以用來修飾對象,那么我們?yōu)槭裁床挥盟夭龋恳驗楸籥ssign修飾的對象在釋放之后,指針的地址還是存在的霍殴,也就是說指針并沒有被置為nil媒惕。如果在后續(xù)的內(nèi)存分配中,剛好分到了這塊地址来庭,程序就會崩潰掉妒蔚。

而weak修飾的對象在釋放之后,指針地址會被置為nil。

__block vs __weak

__block:使用__block修飾的變量可以在__block中被修改肴盏,且會被retain(MRC下不會retain)

__weak:使用__weak修飾的變量不會在block代碼塊中被retain

同時科盛,在ARC下,要避免block出現(xiàn)循環(huán)引用要用:

__weak__typeof(&*self)weakSelf =self;

等同于__weakUIViewController*weakSelf =self;

為什么不用block 是因為通過引用來訪問self的實例變量 菜皂,self被retain,block也是一個強引用贞绵,引起循環(huán)引用,用week是弱引用幌墓,當(dāng)self釋放時但壮,weakSelf已經(jīng)等于nil。

MRC模式下
使用__block能夠避免引起循環(huán)引用的問題

ARC模式下

使用__unsafe_unretained?和 __weak都可以避免循環(huán)引用的問題常侣,但由于前者是unsafe的蜡饵,會造成野指針問題,所以盡量少用unsafe_unretained關(guān)鍵字

另外在多線程環(huán)境下(block中的wSelf有可能被析構(gòu)的情況下)胳施,需要先將self轉(zhuǎn)為strong指針溯祸,避免在運行到某個關(guān)鍵步驟時self對象被析構(gòu)。

可參考AFNetworking代碼:

__weak__typeof(self)weakSelf =self;

AFNetworkReachabilityStatusBlock callback =? ^(AFNetworkReachabilityStatus status) { ? ? ? ? ? __strong__typeof(weakSelf)strongSelf = weakSelf;? ?

strongSelf.networkReachabilityStatus= status;

if(strongSelf.networkReachabilityStatusBlock) {? ? ? ?

strongSelf.networkReachabilityStatusBlock(status);? ?

}};

第一行:__weak __typeof(self)weakSelf = self;

為防止callback內(nèi)部對self強引用舞肆,weak一下焦辅。

其中用到了__typeof(self),這里涉及幾個知識點:

1. __typeof椿胯、__typeof__筷登、typeof的區(qū)別

他們沒有區(qū)別,在早期C語言中沒有typeof這個關(guān)鍵字哩盲,__typeof前方、__typeof__是在C語言的擴展關(guān)鍵字的時候出現(xiàn)的。

typeof是現(xiàn)代GNU C++的關(guān)鍵字廉油,從Objective-C的根源說惠险,他其實來自于C語言,所以AFNetworking使用了繼承自C的關(guān)鍵字抒线。

2. 對于老的LLVM編譯器上面這句話會編譯報錯班巩,所以在很早的ARC使用者中流行__typeof(&*self)這種寫法,原因如下

大致說法是老LLVM編譯器會將__typeof轉(zhuǎn)義為 XXX類名 *const __strong的__strong和前面的__weak關(guān)鍵字對指針的修飾又沖突了嘶炭,所以加上&*對指針的修飾抱慌。

第三行:__strong __typeof(weakSelf)strongSelf = weakSelf;

self轉(zhuǎn)回strong了,這里__typeof()里面寫的是weakSelf旱物,里面寫self也沒有問題遥缕,因為typeof是編譯時確定變量類型,所以這里寫self 不會被循環(huán)引用宵呛。

第四单匣、五、六行,如果不轉(zhuǎn)成strongSelf而使用weakSelf户秤,后面幾句話中码秉,有可能在第四句執(zhí)行之后self的對象可能被析構(gòu)掉,然后后面的StausBlock沒有執(zhí)行鸡号,導(dǎo)致邏輯錯誤转砖。

最后第五行,使用前對block判空鲸伴。


若 object 本身沒有去 retain 這個 block (即沒有把這個 block 作成一個 property)府蔗,則可以直接在 block 中使用 self

dispatch_block_t completionBlock = ^{

? ? // 未 retain block,可直接用 self

? ? NSLog(@"%@", self);

}

MyViewController *myController = [[MyViewController alloc] init...];

[self presentViewController:myController?animated:YES?completion:completionHandler];

若有 retain block汞窗,直接用 self 會造成 retain cycle

self.completionHandler = ^{

? ? // 有 retain block姓赤,直接用 self 會造成 retain cycle

? ? NSLog(@"%@", self);

}

MyViewController *myController = [[MyViewController alloc] init...];

[self presentViewController:myController?animated:YES?completion:self.completionHandler];

當(dāng)有 retain block 時,應(yīng)該使用 weakSelf

__weak typeof(self) weakSelf = self;

self.completionHandler = ^{

? ? // 打破 retain cycle

? ? NSLog(@"%@", weakSelf);

};

MyViewController *myController = [[MyViewController alloc] init...];

[self presentViewController:myController?animated:YES?completion:self.completionHandler];

但只用 weakSelf 的問題在於仲吏,如果在 block 中必須多次使用到 weakSelf 會有危險不铆,因為在多執(zhí)行緒下,weakSelf 有可能在 block 跑到一半的時候被設(shè)成 nil

__weak typeof(self) weakSelf = self;

dispatch_block_t block =? ^{

? ? [weakSelf doSomething]; // weakSelf != nil

? ? // preemption, weakSelf turned nil

? ? [weakSelf doSomethingElse]; // weakSelf == nil

};

因此必須在 block 內(nèi)使用 strongSelf裹唆,確保 reference 不會執(zhí)行到一半變成 nil

__weak typeof(self) weakSelf = self;

myObj.myBlock =? ^{

? ? __strong typeof(self) strongSelf = weakSelf;

? ? if (strongSelf) {

? ? ? ? [strongSelf doSomething]; // strongSelf != nil

? ? ? ? // preemption, strongSelf still not nil

? ? ? ? [strongSelf doSomethingElse]; // strongSelf != nil

? ? }?else {

? ? ? ? // Probably nothing...

? ? ? ? return;

? ? }

};

總結(jié):

1. 當(dāng) block 不是 property 時誓斥,用 self 即可

2. 當(dāng) block 是 property,需使用 weakSelf

3. 當(dāng) block 內(nèi)會多次使用 weakSelf许帐,且有用到多執(zhí)行緒劳坑,需使用 strongSelf

4. 並不是所有 block 都得用 weakSelf?(事實上大多數(shù)的 iOS 原生套件,以及 GCD 的 block 是不會造成 retain cycle 的成畦,因為他們並沒有去 retain block)

此外也可以藉由把 block property 設(shè)為 nil 來打破 retain cycle

例如 AFNetworking 就使用了類似的實作方式

因此在 AFNetworking 的 block 中使用 self 也不會造成 retain cycle 問題

另外即使將變數(shù)直接宣告成 instance variable 而非 property泡垃,在 block 中使用時還是會 retain 到 self 而發(fā)生 retain cycle,因為 ivar 其實也是 self 的一部分

@interface MyViewController () {

? ? NSString *tempStr;

}

self.completionHandler = ^{

? ? // 這裡的 tempStr 相當(dāng)於 self->tempStr羡鸥,因此還是會造成 retain cycle

? ? NSLog(@"%@", tempStr);

}

但 ivar 又無法使用 weakSelf 去取值,因此解決方法有

1. 乖乖建立 property (可能比較簡單)

2. 使用 weakSelf + strongSelf

__weak __typeof(self) weakSelf = self;

self.completionHandler = ^{

? ? // 用 weakSelf->tempStr 是無法取值的

? ? __strong __typeof(weakSelf) strongSelf = weakSelf;

? ? NSLog(@"%@", strongSelf->tempStr);

}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末忠寻,一起剝皮案震驚了整個濱河市惧浴,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌奕剃,老刑警劉巖衷旅,帶你破解...
    沈念sama閱讀 206,126評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異纵朋,居然都是意外死亡柿顶,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評論 2 382
  • 文/潘曉璐 我一進店門操软,熙熙樓的掌柜王于貴愁眉苦臉地迎上來嘁锯,“玉大人,你說我怎么就攤上這事〖页耍” “怎么了蝗羊?”我有些...
    開封第一講書人閱讀 152,445評論 0 341
  • 文/不壞的土叔 我叫張陵,是天一觀的道長仁锯。 經(jīng)常有香客問我耀找,道長,這世上最難降的妖魔是什么业崖? 我笑而不...
    開封第一講書人閱讀 55,185評論 1 278
  • 正文 為了忘掉前任野芒,我火速辦了婚禮,結(jié)果婚禮上双炕,老公的妹妹穿的比我還像新娘狞悲。我一直安慰自己,他們只是感情好雄家,可當(dāng)我...
    茶點故事閱讀 64,178評論 5 371
  • 文/花漫 我一把揭開白布效诅。 她就那樣靜靜地躺著,像睡著了一般趟济。 火紅的嫁衣襯著肌膚如雪乱投。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 48,970評論 1 284
  • 那天顷编,我揣著相機與錄音戚炫,去河邊找鬼。 笑死媳纬,一個胖子當(dāng)著我的面吹牛双肤,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播钮惠,決...
    沈念sama閱讀 38,276評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼茅糜,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了素挽?” 一聲冷哼從身側(cè)響起蔑赘,我...
    開封第一講書人閱讀 36,927評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎预明,沒想到半個月后缩赛,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,400評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡撰糠,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,883評論 2 323
  • 正文 我和宋清朗相戀三年酥馍,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片阅酪。...
    茶點故事閱讀 37,997評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡旨袒,死狀恐怖汁针,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情峦失,我是刑警寧澤扇丛,帶...
    沈念sama閱讀 33,646評論 4 322
  • 正文 年R本政府宣布,位于F島的核電站尉辑,受9級特大地震影響帆精,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜隧魄,卻給世界環(huán)境...
    茶點故事閱讀 39,213評論 3 307
  • 文/蒙蒙 一卓练、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧购啄,春花似錦襟企、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,204評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至几迄,卻和暖如春蔚龙,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背映胁。 一陣腳步聲響...
    開封第一講書人閱讀 31,423評論 1 260
  • 我被黑心中介騙來泰國打工木羹, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人解孙。 一個月前我還...
    沈念sama閱讀 45,423評論 2 352
  • 正文 我出身青樓坑填,卻偏偏與公主長得像,于是被迫代替她去往敵國和親弛姜。 傳聞我的和親對象是個殘疾皇子脐瑰,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,722評論 2 345

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