iOS-NSAutoreleasePool自動釋放原理及詳解

前言:當(dāng)您向一個對象發(fā)送一個autorelease消息時,Cocoa就會將該對象的一個引用放入到最新的自動釋放池。它仍然是個正當(dāng)?shù)膶ο螅虼俗詣俞尫懦?定義的作用域內(nèi)的其它對象可以向它發(fā)送消息。當(dāng)程序執(zhí)行到作用域結(jié)束的位置時,自動釋放池就會被釋放,池中的所有對象也就被釋放。

1. ojc-c 是通過一種"referring counting"(引用計數(shù))的方式來管理內(nèi)存的, 對象在開始分配內(nèi)存(alloc)的時候引用計數(shù)為一,以后每當(dāng)碰到有alloc,new,copy,retain的時候引用計數(shù)都會加一, 每當(dāng)碰到release和autorelease的時候引用計數(shù)就會減一,如果此對象的計數(shù)變?yōu)榱?, 就會被系統(tǒng)銷毀.

2. NSAutoreleasePool 就是用來做引用計數(shù)的管理工作的,這個部分后面會詳細說到.

3. autorelease和release沒什么區(qū)別,只是引用計數(shù)減一的時機不同而已,autorelease會在對象的使用真正結(jié)束的時候才做引用計數(shù)減一.

4.設(shè)定項目編譯環(huán)境為ARC下時,編譯器會幫助我們在程序的入口main函數(shù)就調(diào)用NSAutoreleasePool,這樣保證程序中不調(diào)用NSAutoreleasePool,但在退出時自動釋放

1.NSAutoreleasePool是什么?

NSAutoreleasePool實際上是個對象引用計數(shù)自動處理器,在官方文檔中被稱為是一個類。

NSAutoreleasePool可以同時有多個,它的組織是個棧,總是存在一個棧頂pool,也就是當(dāng)前pool,每創(chuàng)建一個pool,就往棧里壓一個,改變當(dāng)前pool為新建的pool,然后,每次給pool發(fā)送drain消息,就彈出棧頂?shù)膒ool,改當(dāng)前pool為棧里的下一個 pool。

2.NSAutoreleasePool可以用來做什么,怎么用?

NSAutoreleasePool可以在一定程度上幫助我們蘋果開發(fā)程序員管理內(nèi)存,讓我們的工作更加嚴密,簡便。

1)在ARC項目中,系統(tǒng)會自動幫助我們在程序中嵌入NSAutoreleasePool,此為蘋果公司程序員在寫這個編譯器的時候設(shè)定的;

2)在MRC項目中,我們需要自己去創(chuàng)建NSAutoreleasePool類對象去幫助我們管理內(nèi)存;

3)使用應(yīng)注意:

a.在ARC項目中我們同樣可以創(chuàng)建NSAutoreleasePool類對象去幫助我們更精確的管理內(nèi)存問題。

b. NSAutoreleasePool的管理范圍是在NSAutoreleasePool *pool = [[NSAutoreleasePool alloc]init];與[pool release];之間的對象

c..既然ARC項目中設(shè)置了ARC,為什么還要使用@autoreleasepool?(注意a的案例解釋)

ARC 并不是舍棄了@autoreleasepool,而是在編譯階段幫你插入必要的retain/release/autorelease的代碼調(diào)用。

所以,跟你想象的不一樣,ARC 之下依然是延時釋放的,依然是依賴于NSAutoreleasePool,跟非 ARC 模式下手動調(diào)用那些函數(shù)本質(zhì)上毫無差別,只是編譯器來做會保證引用計數(shù)的正確性。

參考:Retain count semantics in ARC

What's @autoreleasepool

d.NSAutoreleasePool *pool=[[NSAutoreleasePool alloc] init];

當(dāng)執(zhí)行[pool autorelease]的時候,系統(tǒng)會進行一次內(nèi)存釋放,把autorelease的對象釋放掉,如果沒有NSAutoreleasePool , 那這些內(nèi)存不會釋放

注意,對象并不是自動被加入到當(dāng)前pool中,而是需要對對象發(fā)送autorelease消息,這樣,對象就被加到當(dāng)前pool的管理里了。當(dāng)當(dāng)前pool接受到drain消息時,它就簡單的對它所管理的所有對象發(fā)送release消息。(如例子1)

e.在ARC項目中.不能直接使用autorelease pools,而是使用@autoreleasepool{},

@autoreleasepool{}比直接使用NSAutoreleasePool效率高。不使用ARC的時候也可以使用(autorelease嵌套)

4)使用例子:

例子1:

NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];

NSString* nsstring;

char* cstring = "Hello CString";

nsstring = [NSString stringWithUTF8String:cstring];

[pool release];(這一行代碼就是在給pool發(fā)送drain消息了)

官方API摘抄翻譯:

3.autorelease的原理是什么?

Autorelease實際上只是把對release的調(diào)用延遲了,對于每一個Autorelease,系統(tǒng)只是把該Object放入了當(dāng) 前的Autorelease pool中,當(dāng)該pool被釋放時,該pool中的所有Object會被調(diào)用Release。

4.!!!autorelease何時釋放?

對于autorelease pool本身,會在如下兩個條件發(fā)生時候被釋放(詳細信息請參見第5條)

1)、手動釋放Autorelease pool

2)、Runloop結(jié)束后自動釋放

對于autorelease pool內(nèi)部的對象在引用計數(shù)的retain == 0的時候釋放。release和autorelease pool 的 drain都會觸發(fā)retain--事件。

5、autorelease釋放的具體原理是什么?

要搞懂具體原理,則要先要搞清楚autorelease何時會創(chuàng)建。

我們的程序在main()調(diào)用的時候會自動調(diào)用一個autorelease,然后在每一個Runloop, 系統(tǒng)會隱式創(chuàng)建一個Autorelease pool,這樣所有的release pool會構(gòu)成一個象CallStack一樣的一個棧式結(jié)構(gòu),在每一個Runloop結(jié)束時,當(dāng)前棧頂?shù)?Autorelease pool(main()里的autorelease)會被銷毀,這樣這個pool里的每個Object會被release。

可以把autorelease pool理解成一個類似父類與子類的關(guān)系,main()創(chuàng)建了父類,每個Runloop自動生成的或者開發(fā)者自定義的autorelease pool都會成為該父類的子類。當(dāng)父類被釋放的時候,沒有被釋放的子類也會被釋放,這樣所有子類中的對象也會收到release消息。

那什么是一個Runloop呢? 一個UI事件,Timer call, delegate call, 一個鼠標事件,鍵盤按下(MAC OSX),或者iphone上的觸摸事件,異步http連接下后當(dāng)接收完數(shù)據(jù)時,都會是一個新的Runloop。

一般來說,消息循環(huán)運行一次是毫秒級甚至微秒級的,因此autorelease的效率仍然是非常高的,確實是一個巧妙的設(shè)計。

6、使用有什么要注意的?

1)、NSAutoreleasePool可以創(chuàng)建一個autorelease pool,但該對象本身也需要被釋放,如:

NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init;

// Code benefitting from a local autorelease pool.

[pool release];

復(fù)制代碼

在引用計數(shù)環(huán)境下,使用[pool release]或[pool drain]效果是相同的,drain僅適用于max os高版本,低版本不適用,而release通用,其它并無太大差別。

2)、在ARC下,不能使用上述方式調(diào)用autorelease,而應(yīng)當(dāng)使用@autoreleasepool,如:

@autoreleasepool {

// Code benefitting from a local autorelease pool.

}

復(fù)制代碼

3)、盡量避免對大內(nèi)存使用該方法,如圖片。對于這種延遲釋放機制,還是盡量少用,最好只用在方法內(nèi)返回小塊內(nèi)存申請地址值的情況下,且參考和領(lǐng)會OC的一些系統(tǒng)方法,如:[NSString stringWithFormat:]。

4)、不要把大量循環(huán)操作放到同一個NSAutoreleasePool之間,這樣會造成內(nèi)存峰值的上升。

7、關(guān)于多線程,有什么要注意的?

我還未實際使用到,在官方API翻譯出類似如下語句:

1)、對于不同線程,應(yīng)當(dāng)創(chuàng)建自己的autorelease pool。如果應(yīng)用長期存在,應(yīng)該定期drain和創(chuàng)建新的autorelease pool

下面這句話摘自官方API,大概是說多線程中如果沒有使用到cocoa的相關(guān)調(diào)用,則不需要創(chuàng)建autorelease pool,我一直沒有理解透徹

If, however, your detached thread does not make Cocoa calls, you do not need to create an autorelease pool.

2)、如果不是使用的NSThread,就不要用aoturelease pool,除非你是多線程模式(multithreading mode) ,可以使用NSThread的isMultiThreaded方法測試你的應(yīng)用是否是多線程模式

PS:

我把它理解為:新開線程最好實現(xiàn)NSAutoreleasePool(當(dāng) 我們點擊一個App中的一個按鈕或者其他可以觸碰開啟新業(yè)務(wù)的UI控件,在程序里面就會自動開啟一條新線程,當(dāng)我們不用這個業(yè)務(wù)的時候,就需要程序幫我們 提前在“程序的主窗口的所有代碼編譯結(jié)束后”之前關(guān)閉這條線程以優(yōu)化線程的使用,一定程度上盡量避免線程開啟太多占用CPU嚴重引起的卡頓問題)(業(yè)務(wù)邏 輯處理-業(yè)務(wù)線程優(yōu)化)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,763評論 6 539
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,238評論 3 428
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,823評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,604評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 72,339評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,713評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,712評論 3 445
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,893評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,448評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 41,201評論 3 357
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,397評論 1 372
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,944評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,631評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,033評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,321評論 1 293
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,128評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 48,347評論 2 377

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