ARC / Autoreleasepool

參考:http://www.lxweimin.com/p/1b66c4d47cd7

在每一個Runloop中,系統會隱式創建一個Autoreleasepool;
(一個UI事件,Timer call, delegate call, 一個鼠標事件,鍵盤按下(MAC OSX),或者iphone上的觸摸事件,異步http連接下后當接收完數據時,都會創建autoreleasepool???)

問題

  • auto release 對象什么時候釋放?
    參考:http://blog.sunnyxx.com/2014/10/15/behind-autorelease/
    在沒有加入autorelease pool情況下,autorelease對象是在當前的runloop迭代結束時釋放的;

  • autoreleasepool中的對象,release將延遲到什么時候執行?
    1.當前autoreleasepool結束后,該pool中所有對象會被release;
    2.手動調用drain釋放;

  • 系統是什么時候釋放的?
    "The Application Kit creates an autorelease pool on the main thread at the beginning of every cycle of the event loop".
    在每一個事件周期的開始,系統會自動創建一個自動釋放池,在每一個事件周期的結尾,系統會自動銷毀這個自動釋放池;
    ARC之前,會在下一輪runloop時,釋放這個auto-release pool?

  • autoreleasepool 何時需要手動創建?
    1.當你在主線程開啟其他線程;//大多數情況下不需要自己創建線程,可以使用系統的GCD和NSOperation;
    2.在短時間內制造了大量自動釋放對象時,及時銷毀有助于內存資源合理利用;

  • ARC 中 autoreleasepool
    ARC并不是舍棄了autoreleasepool,而是在編譯階段幫你插入必要的retain/release/autorelease的代碼調用;
    ps: 對象并不是自動被加入到當前pool中,而是需要對象發送autorelease消息;

  • ARC下,_weak指向對象被釋放了,那么_weak會被自動設置為nil,那么runtime是如何做到這一點的那?
    在內存中維護一張weak表

{
    id __weak obj1 = obj;
}
/* 編譯器模擬代碼 */
id obj1;
objc_initWeak(&obj1, obj);
objc_destroyWeak(&obj1);

一個對象地址可以對應很多個_weak變量地址,當一個對象被析夠時,那么他在weak表中所指向的_weak變量就會被置為nil,然后再weak表中刪除該記錄;
由于使用weak變量會造成以上系統開銷,所以要僅在需要時才使用;

  • ARC下如何獲取retainCount
    使用_objc_rootRetainCount(obj);

實現原理

詳情可以參考autoreleasepool的objc源碼
@autoreleasepool {} 實際上是兩個函數的調用:
objc_autoreleasePoolPush
objc_autoreleasePoolPop
這兩個函數都調用了AutoreleasePoolPage類中的方法,也就是說他們是通過AutoreleasepoolPage實現的;
閱讀源碼可知,AuoreleasePoolPage的定義中有如下屬性:

 magic_t const magic; //校驗結構完整
 id *next; //指向棧頂
 pthread_t const thread; //指向當前線程
 AutoreleasePoolPage * const parent; //指向父節點
 AutoreleasePoolPage *child; //指向子節點
 uint32_t const depth; //鏈表的深度,即鏈表節點的個數
 uint32_t hiwat; //hight water mark 最高水位標記,棧對象做多時候的個數

這個類有parent跟child節點,可以推斷出這個一個雙向鏈表的節點,而實際上Autoreleasepool的內存結構就是一個雙向鏈表;
也就是說一個線程的autoreleasepool就是一個指針棧,棧中存放的指針指向加入其中的需要release的對象或者POOL_SENINEL(哨兵對象, 用不分割autoreleasepool);
哨兵對象是在入棧時加入的一個autoreleasepool的標記,當autoreleasepool進行出棧時,每一個比這個哨兵對象后進棧的對象都會release;
這個棧是由一個以page為節點的雙向鏈表組成,page根據需求進行增減;若是一個page已存滿autorelease對象,則新建一個page;
autorelease函數和push函數一樣,都是通過autoreleaseFast函數想自動釋放池中添加一個對象,不過push函數的入棧是一個哨兵對象,而autoreleas函數入棧的是需要加入autoreleasepool的對象;

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容