Autorelease 制是 iOS 提供延遲釋放對象的一種機(jī)制逗嫡,放棄對象所有權(quán),但又不想對象會給立即釋放.這個時候 Autorelease 就能為你解決這個問題恕酸,他可以當(dāng)做寄存處,你可以把對象寄存在它那兒,然后在合適的時間去釋放寄存對象芽丹。下面讓我們深入
Autorelease。
從添加對象開始
在 MRC 時代卜朗,可以使用 [xxx autorelease]拔第,來添加一個對象到一個 Autorelease pool 中,什么是 Autorelease pool 场钉? Autorelease pool 是對象池蚊俺,統(tǒng)一管理 autorelease 對象的地方。
在 ARC 時代蘋果禁用了 autorelease 方法逛万,不能用了泳猬,不怕,我們還有 @autorelease{}
可以使用。
為什么我們要添加 autorelease 對象暂殖,我們可以舉一個很常見的例子:你需要在一個函數(shù)中處理大量零碎的對象价匠,然而函數(shù)結(jié)束之后才會釋放這些對象,這個就可以包裹一個 @autorelease{}
可以有效的控制內(nèi)存的占用呛每。
AutoreleasePool 結(jié)構(gòu)
在我們寫下 @autorelease{}
的時候踩窖,編譯器為我們轉(zhuǎn)換成
void *context = objc_autoreleasePoolPush();
// 你的代碼
objc_autoreleasePoolPop(context);
這2個函數(shù)在做什么呢?實(shí)際都是對AutoreleasePoolPage
的封裝
void *objc_autoreleasePoolPush(void) {
return AutoreleasePoolPage::push();
}
void objc_autoreleasePoolPop(void *ctxt) {
AutoreleasePoolPage::pop(ctxt);
}
AutoreleasePoolPage 對象結(jié)構(gòu)如下圖
我們可以注意到child
和parent
2個字段,可以看出AutoreleasePool
是由多個 AutoreleasePoolPage
組成的雙向列表
AutoreleasePool Push
每個AutoreleasePoolPage
的大小都是 4096 字節(jié)晨横,除去 7 字節(jié)存儲成員變量洋腮,其他部分都是用來存放 Autorelease 對象的指針。在 AutoreleasePool Push
的時候系統(tǒng)會初始化一個AutoreleasePoolPage
手形,填充相應(yīng)的字段啥供,首先填入一個哨兵對象。
哨兵對象定義為:
#define POOL_SENTINEL nil
為什么要最先添加哨兵對象呢库糠?哨兵對象也是 nil 伙狐,在push 的時候添加到棧頂是為了在 Pop (release)的時候,會連續(xù) Pop (release)到第一個哨兵對象瞬欧。
添加完哨兵對象之后就將 next 指針指向下一個添加的 Autorelease 對象贷屎。以此加入對象,如果一個AutoreleasePoolPage
被用慢艘虎,那么會開啟一個新的AutoreleasePoolPage
唉侄,并且將前一個的child指針指向新的AutoreleasePoolPage
,添加雙向鏈表節(jié)點(diǎn)野建。
AutoreleasePool Pop
Pop 操作會以此釋放棧頂對象属划,會一直釋放對象到一個哨兵對象,也就是上文說的候生。
當(dāng)然 Pop 也支持 Pop 到指定對象同眯。
Autoreleasepool 與 Runloop 的關(guān)系
Autoreleasepool 處理了 RunLoop 的3個事件
- Entry(即將進(jìn)入Loop),這個時候會調(diào)用 _objc_autoreleasePoolPop 來創(chuàng)建一個新的釋放池陶舞,
- BeforeWaiting(準(zhǔn)備進(jìn)入休眠) 時調(diào)用_objc_autoreleasePoolPop() 和 _objc_autoreleasePoolPush() 釋放舊的池并創(chuàng)建新池嗽测。
- Exit(即將退出Loop) 時調(diào)用 _objc_autoreleasePoolPop() 來釋放自動釋放池
看到一些有趣的問題。
什么對象自動加入到 autoreleasepool中?
- 非 alloc/new/copy/mutableCopy 的方法生成的對象如
[NSArray array]
肿孵。 - __weak修飾符 修飾的對象唠粥。
- id的指針或?qū)ο蟮闹羔樤跊]有顯式指定時會被附加上__autorealeasing修飾符
子線程默認(rèn)不會開啟 Runloop,那出現(xiàn) Autorelease 對象如何處理停做?不手動處理會內(nèi)存泄漏嗎晤愧?
子線程如果沒有創(chuàng)建 Pool ,但是產(chǎn)生了 Autorelease 對象蛉腌,就會調(diào)用 autoreleaseNoPage 方法官份。在這個方法中只厘,會自動幫你創(chuàng)建一個 hotpage,也就是默認(rèn)生成一個 AutoreleasePoolPage
來添加 autorelease 對象舅巷。