30.@autoreleasepool的使用(面試點:在for循環(huán)中創(chuàng)建大量局部變量享完,會導(dǎo)致大量內(nèi)存暴增,這個時候用autoreleasepool可以減輕內(nèi)存增長)

自動釋放池block废膘,蘋果官方文檔:Using Autorelease Pool Blocks

面試經(jīng)常會有這樣的問題:

1.什么是@autoreleasepool?

NSAutoreleasePool實際上是個對象引用計數(shù)自動處理器竹海,在官方文檔中被稱為是一個類。Objective-C的對象(全部繼承自NSObject)丐黄,就是使用引用計數(shù)的方法來管理對象的存活斋配,眾所周知,當引用計數(shù)為0時灌闺,對象就被銷毀了艰争。操作非常簡單,當對象被創(chuàng)建時桂对,引用計數(shù)被設(shè)成1甩卓。可以給對象發(fā)送retain消息蕉斜,讓對象對自己的引用計數(shù)加1逾柿。而當對象接受到release消息時缀棍,對象就會對自己的引用計數(shù)進行減1,當引用計數(shù)到了0机错,對象就會調(diào)用自己的dealloc處理爬范。當對象被加入到NSAutoreleasePool中,會對其對象retain一次弱匪,當NSAutoreleasePool結(jié)束時青瀑,會對其所有對象發(fā)送一次release消息。NSAutoreleasePool可以同時有多個萧诫,它的組織是個棧斥难,總是存在一個棧頂pool,也就是當前pool帘饶,每創(chuàng)建一個pool蘸炸,就往棧里壓一個,改變當前pool為新建的pool尖奔,然后搭儒,每次給pool發(fā)送drain消息,就彈出棧頂?shù)膒ool提茁,改當前pool為棧里的下一個pool淹禾。

2.里面對象的內(nèi)存什么時候釋放?

3.什么時候要用@autoreleasepool?

回答:

1.@autoreleasepool是自動釋放池,讓我們更自由的管理內(nèi)存

2.當我們手動創(chuàng)建了一個@autoreleasepool,里面創(chuàng)建了很多臨時變量,當@autoreleasepool結(jié)束時江咳,里面的內(nèi)存就會回收

3.ARC時代,系統(tǒng)自動管理自己的autoreleasepool毁习,runloop就是iOS中的消息循環(huán)機制,當一個runloop結(jié)束時系統(tǒng)才會一次性清理掉被autorelease處理過的對象卖丸,其實本質(zhì)上說是在本次runloop迭代結(jié)束時清理掉被本次迭代期間被放到autorelease pool中的對象的纺且。至于何時runloop結(jié)束并沒有固定的duration。

第一個監(jiān)聽的是activities 是 NSRunLoopEntry狀態(tài)稍浆,說明當runloop進入entry狀態(tài)的時候载碌,會調(diào)用_wrapRunLoopWithAutoreleasePoolHandler
,其內(nèi)部會調(diào)用_objc_autoreleasePoolPush()創(chuàng)建自動釋放池衅枫。
第二個監(jiān)聽的activities是 NSRunLoopBeforeWaiting 和NSRunLoopExit嫁艇,BeforeWaiting 其回調(diào)方法_wrapRunLoopWithAutoreleasePoolHandler內(nèi)部會調(diào)用先調(diào)用pop操作,然后再push 創(chuàng)建一個新的自動釋放池弦撩。Exit會調(diào)用pop操作步咪。

autoreleasepool的執(zhí)行順序就是Entry-->push ---> BeforeWaiting--->pop-->push -->Exit-->pop,按照這樣的順便益楼,保證了猾漫,每一次push都對應(yīng)一個pop纯丸。autoreleasepool釋放操作在每一次runloop 的BeforeWaiting和exit的時候執(zhí)行的

AutoreleasePool并沒有單獨的結(jié)構(gòu),而是由若干個AutoreleasePoolPage以雙向鏈表的形式組合而成(分別對應(yīng)結(jié)構(gòu)中的parent指針和child指針)
AutoreleasePool是按線程一一對應(yīng)的(結(jié)構(gòu)中的thread指針指向當前線程)
AutoreleasePoolPage每個對象會開辟4096字節(jié)內(nèi)存(也就是虛擬內(nèi)存一頁的大芯残洹),除了上面的實例變量所占空間俊扭,剩下的空間全部用來儲存autorelease對象的地址
上面的id *next指針作為游標指向棧頂最新add進來的autorelease對象的下一個位置
一個AutoreleasePoolPage的空間被占滿時队橙,會新建一個AutoreleasePoolPage對象,連接鏈表萨惑,后來的autorelease對象在新的page加入

方便是方便了捐康,但是有些情況下,我們還是需要手動創(chuàng)建自動釋放池庸蔼,那么解总,什么時候呢?

1姐仅,如果你正在編寫不基于UI 框架的程序花枫,比如命令行工具。

2掏膏,如果你編寫的循環(huán)創(chuàng)建了很多臨時對象劳翰。

3,你可以在循環(huán)中使用自動釋放池block馒疹,在下次迭代前處理這些對象佳簸。在循環(huán)中使用自動釋放池block,有助于減少應(yīng)用程序的內(nèi)存占用颖变。

4生均,你生成了一個輔助線程。

一旦線程開始執(zhí)行你必須自己創(chuàng)建自動釋放池腥刹。否則马胧,應(yīng)用將泄漏對象

這是蘋果文檔中的翻譯,按我的理解衔峰,最重要的使用場景漓雅,應(yīng)該是有大量中間臨時變量產(chǎn)生時,避免內(nèi)存使用峰值過高朽色,及時釋放內(nèi)存的場景邻吞。

舉個例子

NSArray *urls = <# An array of file URLs #>;
for (NSURL *url in urls) {
    @autoreleasepool {
        NSError *error;
        NSString *fileContents = [NSString stringWithContentsOfURL:url
                                         encoding:NSUTF8StringEncoding
                                         error:&error];
    }
}

這個for循環(huán)里如果不使用@autoreleasepool,那臨時變量內(nèi)存可能是爆發(fā)式的葫男,但是使用了@autoreleasepool抱冷,在每個@autoreleasepool結(jié)束時,里面的臨時變量都會回收梢褐,內(nèi)存使用更加合理旺遮。

今天在學習大佬博客的時候看到一個問題赵讯,下面代碼會有什么問題?

// largeNumber是一個很大的數(shù)
for (int i = 0; i < largeNumber; i++) {
    NSString *str = [NSString stringWithFormat:@"hello -%04d", i];
    str = [str stringByAppendingString:@" - world"];
    NSLog(@"%@", str);
}

剛開始沒看出什么問題耿眉,就是普通的循環(huán)边翼,每次循環(huán)創(chuàng)建一個局部變量NSString。于是寫了個Demo驗證了下鸣剪,在觀察內(nèi)存的時候發(fā)現(xiàn)了端倪组底,在循環(huán)過程中,內(nèi)存不斷飆升筐骇。

image.png

頓時明白了债鸡,原來問題的關(guān)鍵就是這個largeNumber,當循環(huán)此時很大時铛纬,就會創(chuàng)建大量的局部變量厌均,而且得不到釋放,于是內(nèi)存就爆了告唆。這時候就該@autoreleasepool上場了棺弊,優(yōu)化后代碼:

for (int i = 0; i < largeNumber; i++) {
    @autoreleasepool {
        NSString *str = [NSString stringWithFormat:@"hello -%04d", i];
        str = [str stringByAppendingString:@" - world"];
        NSLog(@"%@", str);
    }
}

查看運行內(nèi)存,可以看到非常平穩(wěn)擒悬。

image.png
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末镊屎,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子茄螃,更是在濱河造成了極大的恐慌缝驳,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,406評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件归苍,死亡現(xiàn)場離奇詭異用狱,居然都是意外死亡,警方通過查閱死者的電腦和手機拼弃,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,732評論 3 393
  • 文/潘曉璐 我一進店門夏伊,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人吻氧,你說我怎么就攤上這事溺忧。” “怎么了盯孙?”我有些...
    開封第一講書人閱讀 163,711評論 0 353
  • 文/不壞的土叔 我叫張陵鲁森,是天一觀的道長。 經(jīng)常有香客問我振惰,道長歌溉,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,380評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮痛垛,結(jié)果婚禮上草慧,老公的妹妹穿的比我還像新娘。我一直安慰自己匙头,他們只是感情好漫谷,可當我...
    茶點故事閱讀 67,432評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著蹂析,像睡著了一般舔示。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上识窿,一...
    開封第一講書人閱讀 51,301評論 1 301
  • 那天,我揣著相機與錄音脑融,去河邊找鬼喻频。 笑死,一個胖子當著我的面吹牛肘迎,可吹牛的內(nèi)容都是我干的甥温。 我是一名探鬼主播,決...
    沈念sama閱讀 40,145評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼妓布,長吁一口氣:“原來是場噩夢啊……” “哼姻蚓!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起匣沼,我...
    開封第一講書人閱讀 39,008評論 0 276
  • 序言:老撾萬榮一對情侶失蹤狰挡,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后释涛,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體加叁,經(jīng)...
    沈念sama閱讀 45,443評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,649評論 3 334
  • 正文 我和宋清朗相戀三年唇撬,在試婚紗的時候發(fā)現(xiàn)自己被綠了它匕。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,795評論 1 347
  • 序言:一個原本活蹦亂跳的男人離奇死亡窖认,死狀恐怖豫柬,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情扑浸,我是刑警寧澤烧给,帶...
    沈念sama閱讀 35,501評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站喝噪,受9級特大地震影響创夜,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜仙逻,卻給世界環(huán)境...
    茶點故事閱讀 41,119評論 3 328
  • 文/蒙蒙 一驰吓、第九天 我趴在偏房一處隱蔽的房頂上張望涧尿。 院中可真熱鬧,春花似錦檬贰、人聲如沸姑廉。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,731評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽桥言。三九已至,卻和暖如春葵礼,著一層夾襖步出監(jiān)牢的瞬間号阿,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,865評論 1 269
  • 我被黑心中介騙來泰國打工鸳粉, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留扔涧,地道東北人。 一個月前我還...
    沈念sama閱讀 47,899評論 2 370
  • 正文 我出身青樓届谈,卻偏偏與公主長得像枯夜,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子艰山,可洞房花燭夜當晚...
    茶點故事閱讀 44,724評論 2 354

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