概念的陳述
autoReleasepool是我們iOS開(kāi)發(fā)項(xiàng)目中及其重要的一個(gè)內(nèi)存管理機(jī)制豌蟋,自從我們是用ARC后号醉,我們?cè)僖膊挥瞄_(kāi)啟一個(gè)內(nèi)容分配給某個(gè)對(duì)象颅痊,我們只管做我們開(kāi)發(fā)應(yīng)該做的事情火脉,內(nèi)存的分配和釋放都交給autoReleasepool去處理牵寺,這樣既方便也省事械姻,所以autoReleasepool的概念就不多敘述了妒蛇,自動(dòng)內(nèi)存釋放池,相信只要做iOS開(kāi)發(fā)的都能明白其作用的重要性楷拳。
AutoReleasepool的內(nèi)部結(jié)構(gòu)
magic_t const magic; magic_t const magic;
id *next; id *next;
..... NSString *string49;
NSString *...; NSString *string50;
NSString *...; pthread_t const thread;
NSString *string3; ......
NSString *string2; ......
NSString *string1;
pthread_t const thread;
AutoreleasePoolPage * const
AutoreleasePoolPage *child;
uint32_t const depth;
uint32_t hiwat;
PAGE_MAX_SIZE;
COUNT;
POOL_BOUNDARY;
Page 1 //第一頁(yè) page2//第二頁(yè) ...
1,AutoreleasePool并沒(méi)有單獨(dú)的結(jié)構(gòu)绣夺,而是由若干個(gè)AutoreleasePoolPage以雙向鏈表的形式組成
2,AutoreleasePoolPage每個(gè)對(duì)象會(huì)開(kāi)辟4096字節(jié)內(nèi)存(也就是虛擬內(nèi)存一頁(yè)的大小)欢揖,除了上面的實(shí)例變量所占空間陶耍,剩下的空間全部用來(lái)儲(chǔ)存autorelease對(duì)象的地址
Runloop與AutoReleasepool的關(guān)系
NSLog(@"相關(guān)的關(guān)系 %@",[NSRunLoop currentRunLoop]);
當(dāng)我們?cè)诔绦蛑杏萌罩敬蛴∠嚓P(guān)數(shù)據(jù)的時(shí)候,顯示的日志是
2020-07-30 10:38:49.003092+0800 VVeboTableViewDemo[1522:62613] 相關(guān)的關(guān)系 <CFRunLoop 0x600001ce0300 [0x7fff8062d750]>{wakeup port = 0x1903, stopped = false, ignoreWakeUps = false,
current mode = kCFRunLoopDefaultMode,
common modes = <CFBasicHash 0x600002eba1f0 [0x7fff8062d750]>{type = mutable set, count = 2,
entries =>
0 : <CFString 0x7fff869c52a0 [0x7fff8062d750]>{contents = "UITrackingRunLoopMode"}
2 : <CFString 0x7fff80640a20 [0x7fff8062d750]>{contents = "kCFRunLoopDefaultMode"}
}
,
common mode items = <CFBasicHash 0x600002eba370 [0x7fff8062d750]>{type = mutable set, count = 11,
entries =>
0 : <CFRunLoopSource 0x6000015e8000 [0x7fff8062d750]>{signalled = No, valid = Yes, order = -1, context = <CFRunLoopSource context>{version = 0, info = 0x0, callout = PurpleEventSignalCallback (0x7fff38c3a9b2)}}
1 : <CFRunLoopSource 0x6000015f00c0 [0x7fff8062d750]>{signalled = No, valid = Yes, order = -1, context = <CFRunLoopSource context>{version = 1, info = 0x2e03, callout = PurpleEventCallback (0x7fff38c3a9be)}}
2 : <CFRunLoopSource 0x6000015fc000 [0x7fff8062d750]>{si
但是相關(guān)控制臺(tái)打印的數(shù)據(jù)是
0 : <CFRunLoopSource 0x6000015e8000 [0x7fff8062d750]>{signalled = No, valid = Yes, order = -1, context = <CFRunLoopSource context>{version = 0, info = 0x0, callout = PurpleEventSignalCallback (0x7fff38c3a9b2)}}
1 : <CFRunLoopSource 0x6000015f00c0 [0x7fff8062d750]>{signalled = No, valid = Yes, order = -1, context = <CFRunLoopSource context>{version = 1, info = 0x2e03, callout = PurpleEventCallback (0x7fff38c3a9be)}}
2 : <CFRunLoopSource 0x6000015fc000 [0x7fff8062d750]>{signalled = No, valid = Yes, order = -1, context = <CFRunLoopSource context>{version = 0, info = 0x600001bf84e0, callout = __handleEventQueue (0x7fff493c2e6e)}}
3 : <CFRunLoopObserver 0x6000011fc1e0 [0x7fff8062d750]>{valid = Yes, activities = 0xa0, repeats = Yes, order = 2147483647, callout = _wrapRunLoopWithAutoreleasePoolHandler (0x7fff4931eaa4), context = <CFArray 0x600002edf570 [0x7fff8062d750]>{type = mutable-small, count = 1, values = (
0 : <0x7fbfa580d048>
)}}
4 : <CFRunLoopObserver 0x6000011fc000 [0x7fff8062d750]>{valid = Yes, activities = 0xa0, repeats = Yes, order = 1999000, callout = _beforeCACommitHandler (0x7fff4934fa4f), context = <CFRunLoopObserver context 0x7fbfa4c052c0>}
5 : <CFRunLoopObserver 0x6000011f80a0 [0x7fff8062d750]>{valid = Yes, activities = 0x20, repeats = Yes, order = 0, callout = _UIGestureRecognizerUpdateObserver (0x7fff48eaa99e), context = <CFRunLoopObserver context 0x600000bf8e00>}
6 : <CFRunLoopSource 0x6000015fc0c0 [0x7fff8062d750]>{signalled = No, valid = Yes, order = -2, context = <CFRunLoopSource context>{version = 0, info = 0x600002ea8db0, callout = __handleHIDEventFetcherDrain (0x7fff493c2edd)}}
7 : <CFRunLoopSource 0x6000015e0000 [0x7fff8062d750]>{signalled = Yes, valid = Yes, order = 0, context = <CFRunLoopSource context>{version = 0, info = 0x6000004e0540, callout = FBSSerialQueueRunLoopSourceHandler (0x7fff36d82bd5)}}
8 : <CFRunLoopObserver 0x6000011fc0a0 [0x7fff8062d750]>{valid = Yes, activities = 0xa0, repeats = Yes, order = 2001000, callout = _afterCACommitHandler (0x7fff4934fab8), context = <CFRunLoopObserver context 0x7fbfa4c052c0>}
9 : <CFRunLoopObserver 0x6000011ec320 [0x7fff8062d750]>{valid = Yes, activities = 0xa0, repeats = Yes, order = 2000000, callout = _ZN2CA11Transaction17observer_callbackEP19__CFRunLoopObservermPv (0x7fff2b477daa), context = <CFRunLoopObserver context 0x0>}
12 : <CFRunLoopObserver 0x6000011fc140 [0x7fff8062d750]>{valid = Yes, activities = 0x1, repeats = Yes, order = -2147483647, callout = _wrapRunLoopWithAutoreleasePoolHandler (0x7fff4931eaa4), context = <CFArray 0x600002edf570 [0x7fff8062d750]>{type = mutable-small, count = 1, values = (
0 : <0x7fbfa580d048>
)}}
第一個(gè)監(jiān)聽(tīng)的是activities 是 NSRunLoopEntry狀態(tài)她混,說(shuō)明當(dāng)runloop進(jìn)入entry狀態(tài)的時(shí)候烈钞,會(huì)調(diào)用_wrapRunLoopWithAutoreleasePoolHandler
,其內(nèi)部會(huì)調(diào)用_objc_autoreleasePoolPush()創(chuàng)建自動(dòng)釋放池坤按。
第二個(gè)監(jiān)聽(tīng)的activities是 NSRunLoopBeforeWaiting 和NSRunLoopExit毯欣,BeforeWaiting 其回調(diào)方法_wrapRunLoopWithAutoreleasePoolHandler內(nèi)部會(huì)調(diào)用先調(diào)用pop操作,然后再push 創(chuàng)建一個(gè)新的自動(dòng)釋放池臭脓。Exit會(huì)調(diào)用pop操作酗钞。
autoreleasepool的執(zhí)行順序就是Entry-->push ---> BeforeWaiting--->pop-->push -->Exit-->pop,按照這樣的順便谢鹊,保證了算吩,每一次push都對(duì)應(yīng)一個(gè)pop。autoreleasepool釋放操作在每一次runloop 的BeforeWaiting和exit的時(shí)候執(zhí)行的
AutoreleasePool并沒(méi)有單獨(dú)的結(jié)構(gòu)佃扼,而是由若干個(gè)AutoreleasePoolPage以雙向鏈表的形式組合而成(分別對(duì)應(yīng)結(jié)構(gòu)中的parent指針和child指針)
AutoreleasePool是按線程一一對(duì)應(yīng)的(結(jié)構(gòu)中的thread指針指向當(dāng)前線程)
AutoreleasePoolPage每個(gè)對(duì)象會(huì)開(kāi)辟4096字節(jié)內(nèi)存(也就是虛擬內(nèi)存一頁(yè)的大匈顺病),除了上面的實(shí)例變量所占空間兼耀,剩下的空間全部用來(lái)儲(chǔ)存autorelease對(duì)象的地址
上面的id *next指針作為游標(biāo)指向棧頂最新add進(jìn)來(lái)的autorelease對(duì)象的下一個(gè)位置
一個(gè)AutoreleasePoolPage的空間被占滿時(shí)压昼,會(huì)新建一個(gè)AutoreleasePoolPage對(duì)象求冷,連接鏈表,后來(lái)的autorelease對(duì)象在新的page加入
所以窍霞,若當(dāng)前線程中只有一個(gè)AutoreleasePoolPage對(duì)象匠题,并記錄了很多autorelease對(duì)象地址時(shí)內(nèi)存如下圖:
圖中的情況,這一頁(yè)再加入一個(gè)autorelease對(duì)象就要滿了(也就是next指針馬上指向棧頂)但金,這時(shí)就要執(zhí)行上面說(shuō)的操作韭山,建立下一頁(yè)page對(duì)象,與這一頁(yè)鏈表連接完成后冷溃,新page的next指針被初始化在棧底(begin的位置)钱磅,然后繼續(xù)向棧頂添加新對(duì)象。
所以似枕,向一個(gè)對(duì)象發(fā)送- autorelease消息盖淡,就是將這個(gè)對(duì)象加入到當(dāng)前AutoreleasePoolPage的棧頂next指針指向的位置。
所以runloop 和AutoReleasepool是通過(guò)線程的方式一一對(duì)應(yīng)的
什么情況下需要?jiǎng)?chuàng)建自動(dòng)釋放池凿歼?
其實(shí)自動(dòng)釋放池存在的意義是為了延遲釋放一些對(duì)象褪迟,延遲向?qū)ο蟀l(fā)送release消息。在實(shí)際的開(kāi)發(fā)中答憔,有兩種情況是需要手動(dòng)創(chuàng)建自動(dòng)釋放池的味赃。
1、在多線程中攀唯,因?yàn)樽泳€程中可能會(huì)使用便利構(gòu)造器等方法來(lái)創(chuàng)建對(duì)象(類方法)洁桌,那么這些對(duì)象的釋放只能放在自動(dòng)釋放池中渴丸,此時(shí)需要在子線程中添加自動(dòng)釋放池侯嘀。
使用便利構(gòu)造器等方法來(lái)創(chuàng)建對(duì)象是autorelease的對(duì)象,需要制動(dòng)釋放池才能釋放
主線程已經(jīng)添加過(guò)自動(dòng)釋放池谱轨,在main函數(shù)里面戒幔。
2、就是如果一段代碼里面(比如for循環(huán))大量使用便利構(gòu)造器創(chuàng)建對(duì)象土童,也需要手動(dòng)添加自動(dòng)釋放池诗茎。
什么是便利構(gòu)造器?
便利構(gòu)造器在初始化方法的基礎(chǔ)上前進(jìn)了一?小步献汗。封裝了對(duì)象創(chuàng)建過(guò)程敢订。
便利構(gòu)造器是“+”方法,返回本類型的實(shí)例,方法名以類名開(kāi)頭,其實(shí)一些類中的靜態(tài)方法也是便利構(gòu)造器罢吃。
可以有0到多個(gè)參數(shù)楚午。內(nèi)部實(shí)現(xiàn):封裝了alloc和初始化方法。使用起來(lái)更加簡(jiǎn)潔尿招。
runloop和 autorelease pool
先提出一個(gè)問(wèn)題矾柜,在Iphone項(xiàng)目中阱驾,大家會(huì)看到一個(gè)默認(rèn)的Autorelease pool,程序開(kāi)始時(shí)創(chuàng)建怪蔑,程序退出時(shí)銷毀里覆,按照對(duì)Autorelease的理解,豈不是所有autorelease pool里的對(duì)象在程序退出時(shí)才release缆瓣, 這樣跟內(nèi)存泄露有什么區(qū)別喧枷?結(jié)果是,對(duì)于每一個(gè)Runloop弓坞, 系統(tǒng)會(huì)隱式創(chuàng)建一個(gè)Autorelease pool割去,這樣所有的release pool會(huì)構(gòu)成一個(gè)象CallStack一樣的一個(gè)棧式結(jié)構(gòu),在每一個(gè)Runloop結(jié)束時(shí)昼丑,當(dāng)前棧頂?shù)腁utorelease pool會(huì)被銷毀呻逆,這樣這個(gè)pool里的每個(gè)Object會(huì)被release。
那什么是一個(gè)runloop菩帝?一個(gè)UI事件咖城,一個(gè)timer,一個(gè)系統(tǒng)delegate都稱之為runloop(不是NSRunloop)呼奢,runloop實(shí)際上是從接收消息宜雀,然后處理完消息的一個(gè)完整過(guò)程。
為了更加形象說(shuō)明auto release pool機(jī)制握础,下面舉例:
NSString* str1是assign辐董。
UI事件:UIButton的target-action機(jī)制,在action中創(chuàng)建一個(gè)autorelease的UILabel對(duì)象禀综,并賦值简烘,在action中打印出值,action執(zhí)行完畢定枷,這個(gè)時(shí)候runloop結(jié)束孤澎,autorelease pool被釋放,label也被釋放欠窒,所以再調(diào)用這個(gè)對(duì)象的值時(shí)覆旭,出現(xiàn)bad_exec_access。