自動釋放池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)存不斷飆升筐骇。
頓時明白了债鸡,原來問題的關(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)擒悬。