1.引用計(jì)數(shù)的思考方式
創(chuàng)新互聯(lián)IDC提供業(yè)務(wù):雅安服務(wù)器托管,成都服務(wù)器租用,雅安服務(wù)器托管,重慶服務(wù)器租用等四川省內(nèi)主機(jī)托管與主機(jī)租用業(yè)務(wù);數(shù)據(jù)中心含:雙線機(jī)房,BGP機(jī)房,電信機(jī)房,移動(dòng)機(jī)房,聯(lián)通機(jī)房。
2.引用計(jì)數(shù)的實(shí)現(xiàn)
1.alloc方法
+ alloc + allocWithZone: class_creatInstance calloc
調(diào)用alloc方法首先調(diào)用allocWithZone:類方法,然后調(diào)用class_creatInstance函數(shù),最后調(diào)用calloc來分配內(nèi)存塊。
2.ratainCount/retain/release 方法
- retainCount __CFDoExternRefOperation CFBasicHashGetCountOfKey
- retain __CFDoExternRefOperation CFBasicHashAddValue
-retainCount __CFDoExternRefOperation CFBasicHashRemoveValue //CFBasicHashRemoveValue 為0時(shí),-release調(diào)用dealloc
各個(gè)方法都通過同一個(gè)__CFDoExternRefOperation函數(shù),調(diào)用一系列名稱相似的函數(shù)。并且從函數(shù)名看出蘋果采用散列表(引用計(jì)數(shù)表)來管理引用計(jì)數(shù),表鍵值為內(nèi)存塊地址的散列值。然而GNUStep將引用計(jì)數(shù)保存在對(duì)象占用內(nèi)存塊頭部的變量中(objc_layout這個(gè)結(jié)構(gòu)體中)。
內(nèi)存塊頭部管理引用計(jì)數(shù)的好處:
引用技術(shù)表管理引用計(jì)數(shù)的好處:
1. 對(duì)象內(nèi)存快的分配無需考慮內(nèi)存塊頭部
引用計(jì)數(shù)表各記錄中存有內(nèi)存塊地址,可從各個(gè)記錄追溯到各個(gè)內(nèi)存塊。
第二條特征在調(diào)試時(shí)很重要,即使出現(xiàn)故障導(dǎo)致對(duì)象占用的內(nèi)存塊損壞,但只要引用計(jì)數(shù)表沒有被損壞,就能夠確認(rèn)各個(gè)內(nèi)存塊的地址
3.autorelease方法
NSAutoreleasePool是通過以AutoreleasePoolPage為結(jié)點(diǎn)的雙向鏈表來實(shí)現(xiàn)的。AutoreleasePoolPage是一個(gè)C++實(shí)現(xiàn)的類,類結(jié)構(gòu)如圖:
AutoreleasePoolPage每個(gè)對(duì)象會(huì)開辟4096字節(jié)內(nèi)存(也就是虛擬內(nèi)存一頁(yè)的大?。?,除了實(shí)例變量所占空間,剩下的空間全部用來儲(chǔ)存autorelease對(duì)象的地址。內(nèi)存結(jié)構(gòu)如圖:
在Cocoa框架中,NSRunloop每次循環(huán)過程中NSAutoreleasePool對(duì)象被生成或廢棄。在大量產(chǎn)生autorelease對(duì)象時(shí),只要不廢棄NSAutoreleasePool那么生成的對(duì)象就不能被釋放,在此情況下有時(shí)會(huì)產(chǎn)生內(nèi)存不足的現(xiàn)象,因此有必要適當(dāng)?shù)纳?,持有和廢棄NSAutoreleasePool。通常在使用Objective-C,無論調(diào)用哪一個(gè)對(duì)象的autorelease/retain方法,實(shí)現(xiàn)上都是調(diào)用NSObject類的autorelease/retain實(shí)例方法,但是對(duì)于NSAutoreleasePool類,autorelease/retain實(shí)例方法已被重寫,因此運(yùn)行時(shí)會(huì)出錯(cuò)(exception)。autorelease實(shí)際上把對(duì)象的釋放時(shí)機(jī)交給NSAutoreleasePool管理,使用方法如下:
生成并持有NSAutoreleasePool對(duì)象。
NSAutoreleasePool *pool = [NSAutoreleasePool alloc] init]; // 等同于 objc_autoreleasePoolPush()
調(diào)用已分配對(duì)象的autorelease實(shí)例方法。
id obj = [NSObject alloc] init]; [obj autorelease]; // 等同于 objc_autorelease()obj
廢棄NSAutoreleasPool對(duì)象(自動(dòng)調(diào)用分配對(duì)象的release)。
[pool drain]; // 等同于 objc_autoreleasePoolPop(pool)
4.ARC說明
ARC(Automatic Reference Counting)是編譯階段自動(dòng)做了retain/release,原先需要手動(dòng)添加處理引用計(jì)數(shù)的代碼可以自動(dòng)地由編譯器完成。ARC并不是GC,不是運(yùn)行時(shí)內(nèi)存管理,不會(huì)做malloc/free的工作,它只是一種代碼靜態(tài)分析(Static Analyzer)工具,同一程序中按文件單位可以選擇ARC有效和無效。Core Foundation中的malloc()或者free()等,還是需要自己手動(dòng)進(jìn)行內(nèi)存管理。設(shè)置ARC有效的編譯方法如下:
3.引用計(jì)數(shù)查看
Apple 提供一些方法查看對(duì)象的引用計(jì)數(shù),但是并不能完全信任這些函數(shù)提供的引用計(jì)數(shù)值。對(duì)于已釋放的對(duì)象一級(jí)不正確的對(duì)象地址,有時(shí) 也返回”1“,在多線程中,因?yàn)榇嬖诟?jìng)態(tài)條件的問題,所以取得的的數(shù)值不一定可信。
[object retainCount]; //得到object的引用計(jì)數(shù),此方法僅僅適用于MRC _objc_rootRetainCount(obj); //MRC和ARC都適用