一、weak基本用法
weak是弱引用瘪撇,用weak來修飾获茬、描述所引用對象的計(jì)數(shù)器并不會加1,而且weak會在引用對象被釋放的時(shí)候自動置為nil倔既,這也就避免了野指針訪問壞內(nèi)存而引起奔潰的情況恕曲,另外weak也可以解決循環(huán)引用。
拓展:為什么修飾代理使用weak而不是用assign叉存?
assign可用來修飾基本數(shù)據(jù)類型码俩,也可修飾OC的對象度帮,但如果用assign修飾對象類型指向的是一個強(qiáng)指針歼捏,當(dāng)指向的這個指針釋放之后,它仍指向這塊內(nèi)存笨篷,必須要手動給置為nil瞳秽,否則會產(chǎn)生野指針,如果還通過此指針操作那塊內(nèi)存率翅,會導(dǎo)致EXC_BAD_ACCESS錯誤练俐,調(diào)用了已經(jīng)被釋放的內(nèi)存空間;而weak只能用來修飾OC對象冕臭,而且相比assign比較安全腺晾,如果指向的對象消失了燕锥,那么它會自動置為nil,不會導(dǎo)致野指針悯蝉。
二归形、weak原理概括
- weak表其實(shí)是一個哈希表,key是所指對象的指針鼻由,value是weak指針的地址數(shù)組暇榴。(value是數(shù)組的原因是:因?yàn)橐粋€對象可能被多個弱引用指針指向)
- Runtime維護(hù)了一張weak表,用來存儲某個對象的所有的
weak指針蕉世。
weak原理實(shí)現(xiàn)過程三步驟
初始化開始時(shí)蔼紧,會調(diào)用objc_initWeak函數(shù),初始化新的weak指針指向?qū)ο蟮牡刂?/p>
2.緊接著奸例,objc_initWeak函數(shù)里面會調(diào)用objc_storeWeak() 函數(shù),objc_storeWeak() 函數(shù)的作用是用來更新指針的指向向楼,創(chuàng)建弱引用表哩至。
3.在最后會調(diào)用clearDeallocating函數(shù)。而clearDeallocating函數(shù)首先根據(jù)對象的地址獲取weak指針地址的數(shù)組蜜自,然后緊接著遍歷這個數(shù)組菩貌,將其中的數(shù)組開始置為nil,把這個entry從weak表中刪除重荠,最后一步清理對象的記錄箭阶。詳細(xì)步驟
- 初始化開始時(shí),會調(diào)用objc_initWeak函數(shù)戈鲁,初始化新的weak指針指向?qū)ο蟮牡刂?br> 當(dāng)我們初始化weak變量時(shí)仇参,runtime會調(diào)用NSObject.mm中的objc_initWeak,而objc_initWeak函數(shù)里面的實(shí)現(xiàn)如下:
id objc_initWeak(id *location, id newObj) { // 查看對象實(shí)例是否有效,無效對象直接導(dǎo)致指針釋放 if (!newObj) { *location = nil; return nil; } // 這里傳遞了三個 bool 數(shù)值 // 使用 template 進(jìn)行常量參數(shù)傳遞是為了優(yōu)化性能 return storeWeakfalse/*old*/, true/*new*/, true/*crash*/> (location, (objc_object*)newObj); }
通過上面代碼可以看出婆殿,objc_initWeak()函數(shù)首先判斷指針指向的類對象是否有效诈乒,無效,直接返回婆芦;否則通過storeWeak()被注冊為一個指向value的_weak對象
- objc_initWeak函數(shù)里面會調(diào)用objc_storeWeak() 函數(shù)怕磨,objc_storeWeak() 函數(shù)的作用是用來更新指針的指向,創(chuàng)建弱引用表肠鲫。
- 在最后會調(diào)用clearDeallocating函數(shù)。而clearDeallocating函數(shù)首先根據(jù)對象的地址獲取weak指針地址的數(shù)組导饲,然后緊接著遍歷這個數(shù)組,將其中的數(shù)組開始置為nil贱除,把這個entry從weak表中刪除悬蔽,最后一步清理對象的記錄。
三始藕、當(dāng)weak指向的對象被釋放時(shí),如何讓weak指針置為nil的呢椒惨?
1缤至、調(diào)用objc_release 2、因?yàn)閷ο蟮囊糜?jì)數(shù)為0,所以執(zhí)行dealloc 3领斥、在dealloc中嫉到,調(diào)用了_objc_rootDealloc函數(shù) 4、在_objc_rootDealloc中月洛,調(diào)用了object_dispose函數(shù) 5何恶、調(diào)用objc_destructInstance 6、最后調(diào)用objc_clear_deallocating,詳細(xì)過程如下: a. 從weak表中獲取廢棄對象的地址為鍵值的記錄 b. 將包含在記錄中的所有附有 weak修飾符變量的地址嚼黔,賦值為nil c. 將weak表中該記錄刪除 d. 從引用計(jì)數(shù)表中刪除廢棄對象的地址為鍵值的記錄