參考
Autorelease Pool是什么
int main(int argc, char * argv[]) {
@autoreleasepool {
return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
}
}
新建一個工程,總會見到這樣的幾行代碼狠鸳,這行代碼將所有的事件娇澎、消息全部交給了UIApplication
來處理据悔。
同時也說明,整個iOS應用洒敏,是包含在一個自動釋放池block中的科乎。
編譯的時候宴杀,這段代碼會被轉(zhuǎn)換成
{
__AtAutoreleasePool __autoreleasepool;
}
其中,出現(xiàn)的結(jié)構體__AtAutoreleasePool
為
struct __AtAutoreleasePool {
__AtAutoreleasePool() {atautoreleasepoolobj = objc_autoreleasePoolPush();}
~__AtAutoreleasePool() {objc_autoreleasePoolPop(atautoreleasepoolobj);}
void * atautoreleasepoolobj;
};
這表明蜗元,我們的main
函數(shù)實際執(zhí)行了
int main(int argc, const char * argv[]) {
{
void * atautoreleasepoolobj = objc_autoreleasePoolPush();
// do whatever you want
objc_autoreleasePoolPop(atautoreleasepoolobj);
}
return 0;
}
結(jié)論 @autoreleasepool
只是幫助我們少寫了這兩行代碼而已或渤,讓代碼看起來更美觀,然后要根據(jù)上述兩個方法來分析自動釋放池的實現(xiàn)奕扣。
Autorelease Pool的主要結(jié)構
每一個autorelease pool
都是由一系列的AutoreleasePoolPage
組成
class AutoreleasePoolPage {
magic_t const magic; //完整性檢驗
id *next;
pthread_t const thread; //保存當前的進程
AutoreleasePoolPage * const parent; //雙線鏈表使用 指向父節(jié)點
AutoreleasePoolPage *child; //雙向鏈表使用 指向子結(jié)點
uint32_t const depth;
uint32_t hiwat;
};
可見薪鹦,自動釋放池的AutoreleasePoolPage
是以雙向鏈表的結(jié)構連接起來的
而在自動釋放池的內(nèi)存中,AutoreleasePoolPage
被以棧結(jié)構存儲起來惯豆,如下圖
更多詳細實現(xiàn)池磁,請移步自動釋放池的前世今生 ---- 深入解析 Autoreleasepool
MRC與ARC時代的Autorelease
MRC
(Mannul Reference Counting)和ARC
(Automatic Reference Counting),分別對應著手動引用計數(shù)和自動引用計數(shù)楷兽。
對地熄!是計數(shù),不是“ GC芯杀、垃圾回收 ”什么的端考,就是說,在Objective-C
的開發(fā)中揭厚,ARC
不代表像Java
那樣有GC
做垃圾回收却特,所以本質(zhì)上還是要“手動”管理內(nèi)存的。也就是說筛圆,我們在ARC
環(huán)境下寫的代碼裂明,不用自己手動插入“ retain
、release
這些消息 ”太援,ARC
會在編譯時為我們在合適的位置插入闽晦,釋放不必要的內(nèi)存扳碍。
而 @autoreleasepool
就跟對象的release
密切相關。
在MRC
時代尼荆,如果我們想先retain
一個對象左腔,但是并不知道在什么時候可以release
它,我們可以像下面這么做:
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
NSString* str = [[[NSString alloc] initWithString:@"tutuge"] autorelease];
//use str...
[pool release];
//str is released
就是說捅儒,我們可以在創(chuàng)建對象的時候給對象發(fā)送“ autorelease
”消息液样,然后當 NSAutoreleasePool
結(jié)束的時候,“標記過”autorelease
的對象都會被“ release
”掉巧还,也就是會被釋放掉鞭莽。
但是在ARC
時代,我們不用手動發(fā)送autorelease
消息麸祷,ARC
會自動幫我們加澎怒。而這個時候,@autoreleasepool
做的事情阶牍,跟 NSAutoreleasePool
就一模一樣了喷面。
Autorelease對象的釋放時機
實踐內(nèi)容可以參考 你真的懂iOS的autorelease嗎?
在上面引用的文章中,筆者用__weak id
指針指向一個autorelease
對象走孽,在不增加引用的情況下觀察autorelease
對象的釋放情況惧辈。
- 首先在
viewDidLoad
方法中初始化一個Array對象 - 分別嘗試在
viewWillAppear
和viewDidAppeare
方法中觀測Array
的值
發(fā)現(xiàn),在viewWillAppear
方法中磕瓷,對象未被釋放盒齿,而到了viewDidAppear
中就被釋放了,發(fā)現(xiàn)Array
對象并非超出作用域就馬上被釋放困食。得出結(jié)論边翁,autorelease
并不是根據(jù)對象的作用域來決定釋放時機。
實際上硕盹,autorelease
釋放對象的依據(jù)是Runloop
符匾,簡單說,runloop
就是iOS
中的消息循環(huán)機制莱睁,當一個runloop
結(jié)束時系統(tǒng)才會一次性清理掉被autorelease
處理過的對象待讳,其實本質(zhì)上說是在本次runloop
迭代結(jié)束時清理掉被本次迭代期間被放到autorelease pool
中的對象的。至于何時runloop
結(jié)束并沒有固定的duration仰剿。
擴展
既然由runloop
來決定對象釋放時機而不是作用域创淡,那么,在一個{}內(nèi)使用循環(huán)大量創(chuàng)建對象就有可能帶來內(nèi)存上的問題南吮,大量對象會被創(chuàng)建而沒有及時釋放琳彩,這時候就需要靠我們?nèi)斯さ母深Aautorelease
的釋放了。
上文有提到autorelease pool
,一旦一個對象被autorelease
露乏,則該對象會被放到iOS的一個池:autorelease pool
碧浊,其實這個pool
本質(zhì)上是一個stack
,扔到pool
中的對象等價于入棧瘟仿。我們把需要及時釋放掉的代碼塊放入我們生成的autorelease pool
中箱锐,結(jié)束后清空這個自定義的pool
,主動地讓pool
清空掉劳较,從而達到及時釋放內(nèi)存的目的驹止。以上述圖片處理的例子為例,優(yōu)化如下:
for (int i = 0; i <= 1000; i ++) {
//創(chuàng)建一個自動釋放池
NSAutoreleasePool *pool = [NSAutoreleasePool new];//也可以使用@autoreleasePool{domeSomething}的方式
NSString *filePath = [[NSBundle mainBundle] pathForResource:@"test" ofType:@"PNG"];
UIImage *image = [[UIImage alloc] initWithContentsOfFile:filePath];
UIImage *image2 = [image imageByScalingAndCroppingForSize:CGSizeMake(480, 320)];
[image release];
//將自動釋放池內(nèi)存釋放观蜗,它會同時釋放掉上面代碼中產(chǎn)生的臨時變量image2
[pool drain];
}
這樣臊恋,每次循環(huán)結(jié)束時,可以及時的釋放臨時對象的內(nèi)存墓捻,其中抖仅,對自動釋放池的操作可以用上文提到的方法來替代。
@autoreleasePool{
//domeSomeThing;
}
什么時候用@autoreleasepool
根據(jù) Apple的文檔 砖第,使用場景如下:
- 寫基于命令行的的程序時撤卢,就是沒有UI框架,如
AppKit
等Cocoa
框架時梧兼。 - 寫循環(huán)凸丸,循環(huán)里面包含了大量臨時創(chuàng)建的對象。(本文的例子)
- 創(chuàng)建了新的線程袱院。(非
Cocoa
程序創(chuàng)建線程時才需要) - 長時間在后臺運行的任務。
什么對象會加入Autoreleasepool中
- 使用
alloc
瞭稼、new
忽洛、copy
、mutableCopy
的方法進行初始化時环肘,由系統(tǒng)管理對象欲虚,在適當?shù)奈恢?code>release。 - 使用
array
會自動將返回值的對象注冊到Autoreleasepool
悔雹。 -
__weak
修飾的對象复哆,為了保證在引用時不被廢棄,會注冊到Autoreleasepool
中腌零。 - id的指針或?qū)ο蟮闹羔樚菡遥跊]有顯示指定時會被注冊到
Autoleasepool
中。