iOS內(nèi)存管理機(jī)制的原理是引用計(jì)數(shù)强胰,引用計(jì)數(shù)簡(jiǎn)單來(lái)說(shuō)就是統(tǒng)計(jì)一塊內(nèi)存的所有權(quán),當(dāng)這塊內(nèi)存被創(chuàng)建出來(lái)的時(shí)候溉愁,它的引用計(jì)數(shù)從0增加到1,表示有一個(gè)對(duì)象或指針持有這塊內(nèi)存饲趋,擁有這塊內(nèi)存的所有權(quán)拐揭,如果這時(shí)候有另外一個(gè)對(duì)象或指針指向這塊內(nèi)存,那么為了表示這個(gè)后來(lái)的對(duì)象或指針對(duì)這塊內(nèi)存的所有權(quán)奕塑,引用計(jì)數(shù)加1變?yōu)?堂污,之后若有一個(gè)對(duì)象或指針不再指向這塊內(nèi)存時(shí),引用計(jì)數(shù)減1龄砰,表示這個(gè)對(duì)象或指針不再擁有這塊內(nèi)存的所有權(quán)盟猖,當(dāng)一塊內(nèi)存的引用計(jì)數(shù)變?yōu)?,表示沒(méi)有任何對(duì)象或指針持有這塊內(nèi)存换棚,系統(tǒng)便會(huì)立刻釋放掉這塊內(nèi)存式镐。
其中在開(kāi)發(fā)時(shí)引用計(jì)數(shù)又分為ARC(自動(dòng)內(nèi)存管理)和MRC(手動(dòng)內(nèi)存管理)。ARC的本質(zhì)其實(shí)就是MRC圃泡,只不過(guò)是系統(tǒng)幫助開(kāi)發(fā)者管理已創(chuàng)建的對(duì)象或內(nèi)存空間碟案,自動(dòng)在系統(tǒng)認(rèn)為合適的時(shí)間和地點(diǎn)釋放掉已經(jīng)失去作用的內(nèi)存空間,原理是一樣的颇蜡。雖然ARC操作起來(lái)很方便价说,不但減少了代碼量,而且降低了內(nèi)存出錯(cuò)的概率风秤,但因?yàn)锳RC不一定會(huì)及時(shí)釋放鳖目,所以程序有時(shí)候可能會(huì)占用內(nèi)存較大。而MRC若做得好缤弦,通過(guò)手動(dòng)管理领迈,及時(shí)釋放掉不需要的內(nèi)存空間,便可保證程序長(zhǎng)時(shí)間運(yùn)行在良好狀態(tài)上碍沐。
在MRC中會(huì)引起引用計(jì)數(shù)變化的關(guān)鍵字有:alloc狸捅,retain,copy累提,release尘喝,autorelease。(strong關(guān)鍵字只用于ARC斋陪,作用等同于retain)
alloc:當(dāng)一個(gè)類的對(duì)象創(chuàng)建朽褪,需要開(kāi)辟內(nèi)存空間的時(shí)候置吓,會(huì)使用alloc,alloc是一個(gè)類方法缔赠,只能用類調(diào)用衍锚,它的作用是開(kāi)辟一塊新的內(nèi)存空間,并使這塊內(nèi)存的引用計(jì)數(shù)從0增加到1嗤堰,注意戴质,是新的內(nèi)存空間,每次用類alloc出來(lái)的都是一塊新的內(nèi)存空間梁棠,與上一次alloc出來(lái)的內(nèi)存空間沒(méi)有必然聯(lián)系置森,而且上一次alloc出來(lái)的內(nèi)存空間仍然存在,不會(huì)被釋放符糊。
retain:retain是一個(gè)實(shí)例方法凫海,只能由對(duì)象調(diào)用,它的作用是使這個(gè)對(duì)象的內(nèi)存空間的引用計(jì)數(shù)加1男娄,并不會(huì)新開(kāi)辟一塊內(nèi)存空間行贪,通常于賦值是調(diào)用,如:
對(duì)象2=[對(duì)象1 retain]模闲;表示對(duì)象2同樣擁有這塊內(nèi)存的所有權(quán)建瘫。若只是簡(jiǎn)單地賦值,如:對(duì)象2=對(duì)象1尸折;那么當(dāng)對(duì)象1的內(nèi)存空間被釋放的時(shí)候啰脚,對(duì)象2便會(huì)成為野指針,再對(duì)對(duì)象2進(jìn)行操作便會(huì)造成內(nèi)存錯(cuò)誤实夹。
copy:copy同樣是一個(gè)實(shí)例方法橄浓,只能由對(duì)象調(diào)用,返回一個(gè)新的對(duì)象亮航,它的作用是復(fù)制一個(gè)對(duì)象到一塊新的內(nèi)存空間上荸实,舊內(nèi)存空間的引用計(jì)數(shù)不會(huì)變化,新的內(nèi)存空間的引用計(jì)數(shù)從0增加到1缴淋,也就是說(shuō)准给,雖然內(nèi)容一樣,但實(shí)質(zhì)上是兩塊內(nèi)存重抖,相當(dāng)于克隆露氮,一個(gè)變成兩個(gè)。其中copy又分為淺拷貝钟沛、深拷貝和真正的深拷貝畔规,淺拷貝只是拷貝地址與retain等同;深拷貝是拷貝內(nèi)容讹剔,會(huì)新開(kāi)辟新內(nèi)存油讯,與retain不一樣;真正的深拷貝是對(duì)于容器類來(lái)說(shuō)的延欠,如數(shù)組類陌兑、字典類和集合類(包括可變和不可變),假設(shè)有一個(gè)數(shù)組類對(duì)象由捎,普通的深拷貝會(huì)開(kāi)辟一塊新內(nèi)存存放這個(gè)對(duì)象兔综,但這個(gè)數(shù)組對(duì)象里面的各個(gè)元素的地址卻沒(méi)有改變也就是說(shuō)數(shù)組元素只是進(jìn)行了retain或者淺拷貝而已,并沒(méi)有創(chuàng)建新的內(nèi)存空間狞玛,而真正的深拷貝软驰,不但數(shù)組對(duì)象本身進(jìn)行了深拷貝,連數(shù)組元素都進(jìn)行了深拷貝心肪,即為各個(gè)數(shù)組元素開(kāi)辟了新的內(nèi)存空間锭亏。
release:release是一個(gè)實(shí)例方法,同樣只能由對(duì)象調(diào)用硬鞍,它的作用是使對(duì)象的內(nèi)存空間的引用計(jì)數(shù)減1慧瘤,若引用計(jì)數(shù)變?yōu)?則系統(tǒng)會(huì)立刻釋放掉這塊內(nèi)存。如果引用計(jì)數(shù)為0的基礎(chǔ)上再調(diào)用release固该,便會(huì)造成過(guò)度釋放锅减,使內(nèi)存崩潰;
autorelease:autorelease是一個(gè)實(shí)例方法伐坏,同樣只能由對(duì)象調(diào)用怔匣,它的作用于release類似,但不是立刻減1桦沉,相當(dāng)于一個(gè)延遲的release每瞒,通常用于方法返回值的釋放,如便利構(gòu)造器永部。autorelease會(huì)在程序走出自動(dòng)釋放池時(shí)執(zhí)行独泞,通常系統(tǒng)會(huì)自動(dòng)生成自動(dòng)釋放池(即使是MRC下),也可以自己設(shè)定自動(dòng)釋放池苔埋,如:
@autoreleasepool{
obj= [[NSObject alloc]init];
[obj autorelease];
}
當(dāng)程序走出“}”時(shí)obj的引用計(jì)數(shù)就會(huì)減1.
除了以上所述的關(guān)鍵字懦砂,還有一些方法會(huì)引起引用計(jì)數(shù)的變化,如UI中父視圖添加组橄、移除子視圖荞膘,導(dǎo)航控制器或視圖控制器推出新的視圖控制器以及返回,容器類(數(shù)組玉工、字典和集合)添加和移除元素羽资。
當(dāng)子視圖添加到父視圖上時(shí),子視圖的引用計(jì)數(shù)加1遵班,移除時(shí)引用計(jì)數(shù)減1屠升,若父視圖引用計(jì)數(shù)變?yōu)?內(nèi)存被釋放潮改,其所有的子視圖都會(huì)被release一次,即引用計(jì)數(shù)減1腹暖,原則上只有這三種情況子視圖的引用計(jì)數(shù)會(huì)發(fā)生變化汇在,其他如父視圖引用計(jì)數(shù)的加減都不會(huì)影響到子視圖。
容器類的情況與視圖類似脏答,添加元素糕殉,該元素引用計(jì)數(shù)加1,移除元素殖告,該元素引用計(jì)數(shù)減1阿蝶,容器引用計(jì)數(shù)變?yōu)?所占用內(nèi)存被釋放,容器所有元素release黄绩,引用計(jì)數(shù)減1羡洁,其他情況下容器本身的引用計(jì)數(shù)變化不會(huì)影響到容器內(nèi)元素的引用計(jì)數(shù)變化。
導(dǎo)航控制器或視圖控制器推出新的視圖控制器會(huì)使被推出的視圖控制器的引用計(jì)數(shù)加1爽丹,該視圖控制器返回的時(shí)候引用計(jì)數(shù)減1焚廊,具體方法如下:
導(dǎo)航控制器推出視圖控制器調(diào)用方法:- (void)pushViewController:(UIViewController *)viewController animated:(BOOL)animated;
返回時(shí)同樣用導(dǎo)航控制器調(diào)用方法:- (UIViewController *)popViewControllerAnimated:(BOOL)animated;
視圖控制器推出視圖控制器調(diào)用方法:- (void)presentViewController:(UIViewController *)viewControllerToPresent animated: (BOOL)flag completion:(void (^)(void))completion
返回時(shí)被推出的視圖控制器調(diào)用方法:- (void)dismissViewControllerAnimated: (BOOL)flag completion: (void (^)(void))completion
應(yīng)注意:當(dāng)一個(gè)對(duì)象的引用計(jì)數(shù)變?yōu)?占用內(nèi)存被釋放時(shí),會(huì)調(diào)用- (void)dealloc方法习劫,所以如果在MRC下自定義類咆瘟,必須在該方法里將該類中屬性關(guān)鍵字設(shè)置為retain或copy的屬性release一次,以免造成內(nèi)存泄露诽里,重寫方法不要忘記在第一行添加[super dealloc]袒餐。