OC內(nèi)存管理
一、基本原理
(一)為什么要進(jìn)行內(nèi)存管理蕴纳。
由于移動(dòng)設(shè)備的內(nèi)存極其有限,所以每個(gè)APP所占的內(nèi)存也是有限制的秒裕,當(dāng)app所占用的內(nèi)存較多時(shí)袱蚓,系統(tǒng)就會(huì)發(fā)出內(nèi)存警告,這時(shí)需要回收一些不需要再繼續(xù)使用的內(nèi)存空間几蜻,比如回收一些不再使用的對(duì)象和變量等。
管理范圍:任何繼承NSObject的對(duì)象体斩,對(duì)其他的基本數(shù)據(jù)類(lèi)型無(wú)效梭稚。
本質(zhì)原因是因?yàn)閷?duì)象和其他數(shù)據(jù)類(lèi)型在系統(tǒng)中的存儲(chǔ)空間不一樣,其它局部變量主要存放于棧中絮吵,而對(duì)象存儲(chǔ)于堆中弧烤,當(dāng)代碼塊結(jié)束時(shí)這個(gè)代碼塊中涉及的所有局部變量會(huì)被回收,指向?qū)ο蟮闹羔樢脖换厥盏徘茫藭r(shí)對(duì)象已經(jīng)沒(méi)有指針指向暇昂,但依然存在于內(nèi)存中莺戒,造成內(nèi)存泄露。
(二)對(duì)象的基本結(jié)構(gòu)
每個(gè)OC對(duì)象都有自己的引用計(jì)數(shù)器急波,是一個(gè)整數(shù)表示對(duì)象被引用的次數(shù)从铲,即現(xiàn)在有多少東西在使用這個(gè)對(duì)象。對(duì)象剛被創(chuàng)建時(shí)澄暮,默認(rèn)計(jì)數(shù)器值為1名段,當(dāng)計(jì)數(shù)器的值變?yōu)?時(shí),則對(duì)象銷(xiāo)毀泣懊。
在每個(gè)OC對(duì)象內(nèi)部伸辟,都專(zhuān)門(mén)有4個(gè)字節(jié)的存儲(chǔ)空間來(lái)存儲(chǔ)引用計(jì)數(shù)器。
(三)引用計(jì)數(shù)器的作用
判斷對(duì)象要不要回收的唯一依據(jù)就是計(jì)數(shù)器是否為0馍刮,若不為0則存在信夫。
(四)操作
給對(duì)象發(fā)送消息,進(jìn)行相應(yīng)的計(jì)數(shù)器操作卡啰。
Retain消息:使計(jì)數(shù)器+1忙迁,改方法返回對(duì)象本身
Release消息:使計(jì)數(shù)器-1(并不代表釋放對(duì)象)
retainCount消息:獲得對(duì)象當(dāng)前的引用計(jì)數(shù)器值
(五) 對(duì)象的銷(xiāo)毀
當(dāng)一個(gè)對(duì)象的引用計(jì)數(shù)器為0時(shí),那么它將被銷(xiāo)毀碎乃,其占用的內(nèi)存被系統(tǒng)回收姊扔。
當(dāng)對(duì)象被銷(xiāo)毀時(shí),系統(tǒng)會(huì)自動(dòng)向?qū)ο蟀l(fā)送一條dealloc消息梅誓,一般會(huì)重寫(xiě)dealloc方法恰梢,在這里釋放相關(guān)的資源,dealloc就像是對(duì)象的“臨終遺言”梗掰。一旦重寫(xiě)了dealloc方法就必須調(diào)用[super dealloc]嵌言,并且放在代碼塊的最后調(diào)用(不能直接調(diào)用dealloc方法)。
一旦對(duì)象被回收了及穗,那么他所占據(jù)的存儲(chǔ)空間就不再可用摧茴,堅(jiān)持使用會(huì)導(dǎo)致程序崩潰(野指針錯(cuò)誤)。
二埂陆、相關(guān)概念和使用注意
野指針錯(cuò)誤:訪問(wèn)了一塊壞的內(nèi)存(已經(jīng)被回收的苛白,不可用的內(nèi)存)。
僵尸對(duì)象:所占內(nèi)存已經(jīng)被回收的對(duì)象焚虱,僵尸對(duì)象不能再被使用购裙。(打開(kāi)僵尸對(duì)象檢測(cè))
空指針:沒(méi)有指向任何東西的指針(存儲(chǔ)的東西是0,null,nil)鹃栽,給空指針發(fā)送消息不會(huì)報(bào)錯(cuò)
注意:不能使用[p retaion]讓僵尸對(duì)象起死復(fù)生躏率。
三、內(nèi)存管理原則
(一)原則
只要還有人在使用某個(gè)對(duì)象,那么這個(gè)對(duì)象就不會(huì)被回收薇芝;
只要你想使用這個(gè)對(duì)象蓬抄,那么就應(yīng)該讓這個(gè)對(duì)象的引用計(jì)數(shù)器+1;
當(dāng)你不想使用這個(gè)對(duì)象時(shí)夯到,應(yīng)該讓對(duì)象的引用計(jì)數(shù)器-1嚷缭;
(二)誰(shuí)創(chuàng)建,誰(shuí)release
(1)如果你通過(guò)alloc,new,copy來(lái)創(chuàng)建了一個(gè)對(duì)象黄娘,那么你就必須調(diào)用release或者autorelease方法
(2)不是你創(chuàng)建的就不用你去負(fù)責(zé)
(三)誰(shuí)retain峭状,誰(shuí)release
只要你調(diào)用了retain,無(wú)論這個(gè)對(duì)象時(shí)如何生成的逼争,你都要調(diào)用release
(四)總結(jié)
有始有終优床,有加就應(yīng)該有減。曾經(jīng)讓某個(gè)對(duì)象計(jì)數(shù)器加1誓焦,就應(yīng)該讓其在最后-1.
四胆敞、內(nèi)存管理代碼規(guī)范
(一)只要調(diào)用了alloc,就必須有release(autorelease)
(二)Set方法的代碼規(guī)范
(1)基本數(shù)據(jù)類(lèi)型:直接復(fù)制
-(void)setAge:(int)age
{
_age=age;
}
(2)OC對(duì)象類(lèi)型
-(void)setCar:(Car *)car
{
//1.先判斷是不是新傳進(jìn)來(lái)的對(duì)象
If(car!=_car)
{
//2 對(duì)舊對(duì)象做一次release
[_car release];//若沒(méi)有舊對(duì)象杂伟,則沒(méi)有影響
//3.對(duì)新對(duì)象做一次retain
_car=[car retain];
}
}
**(三)dealloc方法的代碼規(guī)范**
(1)一定要[super dealloc]移层,而且要放到最后
(2)對(duì)self(當(dāng)前)所擁有的的其他對(duì)象做一次release操作
-(void)dealloc
{
[_car release];
[super dealloc];
}
五、@property的參數(shù)
(1)內(nèi)存管理相關(guān)參數(shù)
Retain:對(duì)對(duì)象release舊值赫粥,retain新值(適用于OC對(duì)象類(lèi)型)
Assign:直接賦值(默認(rèn)观话,適用于非oc對(duì)象類(lèi)型)
Copy:release舊值,copy新值
(2)是否要生成set方法(若為只讀屬性越平,則不生成)
Readonly:只讀频蛔,只會(huì)生成getter的聲明和實(shí)現(xiàn)
Readwrite:默認(rèn)的,同時(shí)生成setter和getter的聲明和實(shí)現(xiàn)
(3)多線程管理(蘋(píng)果在一定程度上屏蔽了多線程操作)
Nonatomic:高性能秦叛,一般使用這個(gè)
Atomic:低性能
(4)Set和get方法的名稱(chēng)
修改set和get方法的名稱(chēng)晦溪,主要用于布爾類(lèi)型。因?yàn)榉祷夭紶栴?lèi)型的方法名一般以is開(kāi)頭挣跋,修改名稱(chēng)一般用在布爾類(lèi)型中的getter三圆。
@propery(setter=setAbc,getter=isRich) BOOL rich;
BOOL b=p.isRich;// 調(diào)用
六、內(nèi)存管理中的循環(huán)引用問(wèn)題以及解決
案例:每個(gè)人有一張身份證避咆,每張身份證對(duì)應(yīng)一個(gè)人舟肉,不能使用#import的方式相互包含,這就形成了循環(huán)引用牌借。
新的關(guān)鍵字:@class 類(lèi)名度气;——解決循環(huán)引用問(wèn)題,提高性能
@class僅僅告訴編譯器膨报,在進(jìn)行編譯的時(shí)候把后面的名字作為一個(gè)類(lèi)來(lái)處理。
(1)@class的作用:聲明一個(gè)類(lèi),告訴編譯器某個(gè)名稱(chēng)是一個(gè)類(lèi)
(2)開(kāi)發(fā)中引用一個(gè)類(lèi)的規(guī)范
1)在.h文件中使用@class來(lái)聲明類(lèi)
2)在.m文件中真正要使用到的時(shí)候现柠,使用#import來(lái)包含類(lèi)中的所有東西
(3)兩端循環(huán)引用的解決方法
一端使用retain,一端使用assign(使用assign的在dealloc中也不用再release)
七院领、Autorelease
(一)基本用法
(1)會(huì)將對(duì)象放到一個(gè)自動(dòng)釋放池中
(2)當(dāng)自動(dòng)釋放池被銷(xiāo)毀時(shí),會(huì)對(duì)池子里的所有對(duì)象做一次release
(3)會(huì)返回對(duì)象本身
(4)調(diào)用完autorelease方法后够吩,對(duì)象的計(jì)數(shù)器不受影響(銷(xiāo)毀時(shí)影響)
(二)好處
(1)不需要再關(guān)心對(duì)象釋放的時(shí)間
(2)不需要再關(guān)心什么時(shí)候調(diào)用release
(三)使用注意
(1)占用內(nèi)存較大的對(duì)象比然,不要隨便使用autorelease,應(yīng)該使用release來(lái)精確控制
(2)占用內(nèi)存較小的對(duì)象使用autorelease周循,沒(méi)有太大的影響
(四)錯(cuò)誤寫(xiě)法
(1)連續(xù)調(diào)用多次autorelease强法,釋放池銷(xiāo)毀時(shí)執(zhí)行兩次release(-1嗎?)
(2)Alloc之后調(diào)用了autorelease湾笛,之后又調(diào)用了release饮怯。
(五)自動(dòng)釋放池
(1)在iOS程序運(yùn)行過(guò)程中,會(huì)創(chuàng)建無(wú)數(shù)個(gè)池子嚎研,這些池子都是以棧結(jié)構(gòu)(先進(jìn)后出)存在的蓖墅。
(2)當(dāng)一個(gè)對(duì)象調(diào)用autorelease時(shí),會(huì)將這個(gè)對(duì)象放到位于棧頂?shù)尼尫懦刂?br>
(六)自動(dòng)釋放池的創(chuàng)建方式
(1)ios 5.0以前的創(chuàng)建方式
NSAutoreleasePool *pool=[[NSAutoreleasePool alloc] init];
[pool release];//[pool drain];用于mac
(2)Ios5.0以后
@autoreleasepool
{//開(kāi)始代表創(chuàng)建自動(dòng)釋放池
·······
}//結(jié)束代表銷(xiāo)毀自動(dòng)釋放池
(七)Autorelease注意
(1)系統(tǒng)自帶的方法中临扮,如果不包含alloc new copy等论矾,則這些方法返回的對(duì)象都是autorelease的,如[NSDate date]杆勇;
(2)開(kāi)發(fā)中經(jīng)常會(huì)寫(xiě)一些類(lèi)方法來(lái)快速創(chuàng)建一個(gè)autorelease對(duì)象贪壳,創(chuàng)建對(duì)象時(shí)不要直接使用類(lèi)名,而是使用self
八蚜退、ARC內(nèi)存管理機(jī)制
(一)ARC的判斷準(zhǔn)則:
只要沒(méi)有強(qiáng)指針指向?qū)ο笕蜓ィ瑢?duì)象就會(huì)被釋放。
(二)指針?lè)诸?lèi):
(1)強(qiáng)指針:默認(rèn)的情況下关霸,所有的指針都是強(qiáng)指針传黄,關(guān)鍵字strong
(2)弱指針:_ _weak關(guān)鍵字修飾的指針
聲明一個(gè)弱指針如下:
_ _weak Person *p;
ARC中,只要弱指針指向的對(duì)象不在了队寇,就直接把弱指針做清空操作膘掰。
_ _weak Person *p=[[Person alloc] init];//不合理,對(duì)象一創(chuàng)建出來(lái)就被釋放掉佳遣,對(duì)象釋放掉后识埋,ARC把指針自動(dòng)清零。
ARC中在property處不再使用retain,而是使用strong零渐,在dealloc中不需要再[super dealloc]窒舟。
@property(nonatomic,strong)Dog *dog;// 意味著生成的成員變量dog是一個(gè)強(qiáng)指針,相當(dāng)于以前的retain诵盼。
如果換成是弱指針惠豺,則換成weak银还,不需要加 _。
(三)ARC的特點(diǎn)總結(jié):
(1)不允許調(diào)用release洁墙,retain蛹疯,retainCount
(2)不允許重寫(xiě)dealloc,但是不允許調(diào)用[super dealloc]
(3)@property的參數(shù):
Strong:相當(dāng)于原來(lái)的retain(適用于OC對(duì)象類(lèi)型),成員變量是強(qiáng)指針
Weak:相當(dāng)于原來(lái)的assign,(適用于oc對(duì)象類(lèi)型)热监,成員變量是弱指針
Assign:適用于非OC對(duì)象類(lèi)型(基礎(chǔ)類(lèi)型)
(四)補(bǔ)充
讓程序兼容ARC和非ARC部分捺弦。轉(zhuǎn)變?yōu)榉茿RC -fno-objc-arc 轉(zhuǎn)變?yōu)锳RC的, -f-objc-arc 孝扛。
ARC也需要考慮循環(huán)引用問(wèn)題:一端使用retain列吼,另一端使用assign。
提示:字符串是特殊的對(duì)象苦始,但不需要使用release手動(dòng)釋放寞钥,這種字符串對(duì)象默認(rèn)就是autorelease的,不用額外的去管內(nèi)存盈简。