weak實(shí)現(xiàn)原理:
- Runtime維護(hù)了一個(gè)weak表没龙,用于存儲(chǔ)指向某個(gè)對(duì)象的所有weak指針僻爽。
- weak表其實(shí)是一個(gè)hash(哈希)表富岳,Key是所指對(duì)象的地址,Value是weak指針的地址(這個(gè)地址的值是所指對(duì)象的地址)數(shù)組垢袱。
初始化時(shí):runtime會(huì)調(diào)用objc_initWeak函數(shù),初始化一個(gè)新的weak指針指向?qū)ο蟮牡刂贰?/p>
添加引用時(shí):objc_initWeak函數(shù)會(huì)調(diào)用 objc_storeWeak() 函數(shù)港柜, objc_storeWeak() 的作用是更新指針指向请契,創(chuàng)建對(duì)應(yīng)的弱引用表咳榜。
釋放時(shí),調(diào)用clearDeallocating函數(shù)爽锥。clearDeallocating函數(shù)首先根據(jù)對(duì)象地址獲取所有weak指針地址的數(shù)組涌韩,然后遍歷這個(gè)數(shù)組把其中的數(shù)據(jù)設(shè)為nil,最后把這個(gè)entry從weak表中刪除氯夷,最后清理對(duì)象的記錄臣樱。
實(shí)現(xiàn)weak后,為什么對(duì)象釋放后會(huì)自動(dòng)為nil
- runtime 對(duì)注冊(cè)的類腮考, 會(huì)進(jìn)行布局雇毫,對(duì)于 weak 對(duì)象會(huì)放入一個(gè) hash 表中。 用 weak 指向的對(duì)象內(nèi)存地址作為 key秸仙,當(dāng)此對(duì)象的引用計(jì)數(shù)為 0 的時(shí)候會(huì) dealloc嘴拢,假如 weak 指向的對(duì)象內(nèi)存地址是 a ,那么就會(huì)以 a 為鍵寂纪, 在這個(gè) weak 表中搜索席吴,找到所有以 a 為鍵的 weak 對(duì)象,從而設(shè)置為 nil 捞蛋。
當(dāng)weak引用指向的對(duì)象被釋放時(shí)孝冒,又是如何去處理weak指針的呢?
調(diào)用objc_release
因?yàn)閷?duì)象的引用計(jì)數(shù)為0拟杉,所以執(zhí)行dealloc
在dealloc中庄涡,調(diào)用了 _objc_rootDealloc 函數(shù)
在 __objc_rootDealloc中,調(diào)用了object_dispose函數(shù)
調(diào)用objc_destructInstance
最后調(diào)用objc_clear_deallocating,詳細(xì)過程如下:
(1)從weak表中獲取廢棄對(duì)象的地址為鍵值的記錄
(2)將包含在記錄中的所有附有 weak修飾符變量的地址搬设,賦值為 nil
(3)將weak表中該記錄刪除
(4)從引用計(jì)數(shù)表中刪除廢棄對(duì)象的地址為鍵值的記錄
iOS block內(nèi)為什么要使用strongSelf
我們知道使用weakSelf的作用是為了防止強(qiáng)循環(huán)引用, 產(chǎn)生不必要的內(nèi)存泄漏問題. 但是為什么在block內(nèi)部還要重新轉(zhuǎn)成strongSelf.
究其原因, 因?yàn)樵赽lock內(nèi)部的weakSelf有可能為為self或者為nil (比如當(dāng)前界面正在加載網(wǎng)絡(luò)數(shù)據(jù), 而此時(shí)用戶關(guān)閉了該界面). 這樣在某些情況下代碼會(huì)崩潰. 所以為了讓self不為nil, 我們?cè)赽lock內(nèi)部將weakSelf轉(zhuǎn)成strongSelf. 因?yàn)閟trongSelf在block內(nèi)部屬于局部變量了穴店,當(dāng)block結(jié)束時(shí), 該strongSelf變量也會(huì)被自動(dòng)釋放., 既避免了循環(huán)引用, 又讓self在block內(nèi)部不為nil.
故為了保證self在block執(zhí)行過程里一直存在拿穴,對(duì)他強(qiáng)引用strongSelf