內(nèi)存管理
1洲愤、內(nèi)存管理范圍:
需要new或者需要alloc颓芭、init等關(guān)鍵字創(chuàng)建的對(duì)象類型,對(duì)于float柬赐,char亡问,double,struct躺率,enum等基本數(shù)據(jù)類型不需要我們?nèi)ス芾硭麄兊膬?nèi)存玛界。(解釋:因?yàn)閷?duì)象類型會(huì)放在堆中,而非對(duì)象類型會(huì)存放在棧中悼吱,棧中的數(shù)據(jù)會(huì)被系統(tǒng)自動(dòng)回收)
2慎框、堆與棧
堆:一般由程序員分配與釋放,如程序員不釋放后添,當(dāng)程序結(jié)束時(shí)笨枯,可能由系統(tǒng)回收,分配方式類似于鏈表遇西;
棧:由操作系統(tǒng)自動(dòng)分配與釋放馅精,一般用于存放形參,局部變量等值粱檀。
例:
int main() {
Person *p = [[Person alloc] init];
}
其中:p存放在棧中洲敢,而Person存放在堆中
3、引用計(jì)數(shù)器
(1)茄蚯、每一個(gè)oc對(duì)象都有一個(gè)自己的引用計(jì)數(shù)器
(2)压彭、他是一個(gè)整數(shù)睦优,占4個(gè)字節(jié)
(3)、含義:對(duì)象被引用的次數(shù)或者有多少東西正在用這個(gè)對(duì)象
(4)壮不、引用計(jì)數(shù)器為0時(shí)汗盘,對(duì)象占用的內(nèi)存會(huì)被系統(tǒng)回收;大于0時(shí)询一,對(duì)象的內(nèi)存不會(huì)被回收隐孽,除非程序退出了;任何一個(gè)對(duì)象健蕊,被創(chuàng)建以后菱阵,引用計(jì)數(shù)器為1(如使用new,alloc绊诲,copy等創(chuàng)建的對(duì)象)送粱。
(5)、引用計(jì)數(shù)器常見(jiàn)操作:
a掂之、retain:引用計(jì)數(shù)器+1
b抗俄、release:引用計(jì)數(shù)器-1(僅代表計(jì)數(shù)器-1,不代表銷毀對(duì)象)
c世舰、retainCount:獲取當(dāng)前計(jì)數(shù)器的值
4动雹、dealloc方法
(1)、當(dāng)對(duì)象的引用計(jì)數(shù)器為0跟压,對(duì)象即將被銷毀時(shí)胰蝠,系統(tǒng)會(huì)給對(duì)象自動(dòng)調(diào)用delloc方法,通過(guò)判斷有沒(méi)有調(diào)用dealloc方法就可以判定對(duì)象有沒(méi)有被銷毀震蒋;
(2)茸塞、dealloc的重寫(xiě):
一般在重寫(xiě)的dealloc方法中釋放資源,且需要調(diào)用[super dealloc]查剖,并且是放在重寫(xiě)的dealloc方法中的最后钾虐;
(3)、一般不能直接調(diào)用dealloc方法笋庄,因?yàn)檎{(diào)用了就會(huì)釋放對(duì)象占用的內(nèi)存效扫,操作不當(dāng),會(huì)造成野指針直砂。
5菌仁、僵尸對(duì)象與野指針
僵尸對(duì)象:如果一個(gè)對(duì)象已經(jīng)被釋放了,那么稱這個(gè)對(duì)象為僵尸對(duì)象
野指針:指向已被釋放的對(duì)象(即僵尸對(duì)象)的指針静暂,我們稱之為野指針济丘。
6、空指針
沒(méi)有指向存儲(chǔ)空間的指針(里面存放的是nil洽蛀,也就是0)
給空指針發(fā)送消息沒(méi)有反應(yīng)闪盔,但是程序不會(huì)報(bào)錯(cuò)也不會(huì)崩潰
未避免野指針弯院,在對(duì)象被銷毀之后,將對(duì)象指向的指針變?yōu)榭罩羔槪ㄔ贛RC中)
7泪掀、內(nèi)存管理說(shuō)明以及retain和assign關(guān)鍵字:
當(dāng)A對(duì)象想使用B對(duì)象時(shí)一定要對(duì)B對(duì)象進(jìn)行一次retain,這樣才能保證A對(duì)象存在B對(duì)象也存在颂碘,換句話說(shuō)就是异赫,無(wú)論什么時(shí)候,A對(duì)象都可以使用B對(duì)象头岔,當(dāng)A對(duì)象release時(shí)塔拳,一定要對(duì)B對(duì)象也進(jìn)行一個(gè)release。
@interface A:NSObject {
B *_b;
}
@end
@implementation
-(void)setB:(B *)b{
[b retain];
_b = b;
}
-(void)delloc {
[_b release];
[super delloc];
}
@end
為避免發(fā)生僵尸對(duì)象與野指針的情況峡竣,set方法最好改寫(xiě)成以下形式:
-(void)setB:(B *)b{
if(b != _b) {
[_b release]; //原數(shù)據(jù)釋放
_b = [b retain]; //新數(shù)據(jù)保存; retain不但會(huì)將引用計(jì)數(shù)+1還會(huì)返回對(duì)象本身
}
}
但是如果每個(gè)屬性都要這么寫(xiě)一遍靠抑,將會(huì)非常繁瑣,所以我們?cè)诼暶鲗傩缘臅r(shí)候可以聲明成這樣:
@property (retain) B *b;
此時(shí)我們可以不用自己去實(shí)現(xiàn)setter和getter方法适掰,他會(huì)自動(dòng)將屬性實(shí)現(xiàn)成類似上面的方法颂碧。
而asign不具備管理對(duì)象的能力,只能適用于基本數(shù)據(jù)類型类浪,所以他說(shuō)實(shí)現(xiàn)的將會(huì)沒(méi)有release和retain的getter和setter方法载城,如下所示:
@property (retain) NSInteger *num;
//實(shí)現(xiàn)的setter方法如下所示
-(void)setNum:(NSInteger)num {
_num = num;
}
總結(jié):assign不會(huì)幫我們生成具備內(nèi)存管理的setter和getter方法,只會(huì)生成默認(rèn)的方法费就,如果在@property后面什么都不寫(xiě)诉瓦,則會(huì)將其視為assign;而retain會(huì)幫我們生成具備內(nèi)存管理的setter和getter方法力细。
8睬澡、atomic與nonatomic:
atomic:線程安全,但是性能很低
nonatomic:線程不安全眠蚂,但是性能相對(duì)會(huì)高很多
9煞聪、MRC與ARC
MRC(手動(dòng)管理內(nèi)存):在MRC的內(nèi)存管理模式下,與對(duì)變量的管理相關(guān)的方法有:retain,release和autorelease河狐。retain和release方法操作的是引用記數(shù)米绕,當(dāng)引用記數(shù)為零時(shí),便自動(dòng)釋放內(nèi)存馋艺。并且可以用NSAutoreleasePool對(duì)象栅干,對(duì)加入自動(dòng)釋放池(autorelease調(diào)用)的變量進(jìn)行管理,當(dāng)drain時(shí)回收內(nèi)存捐祠。
和內(nèi)存管理相關(guān)的方法
1)alloc 引用計(jì)數(shù)器自動(dòng)設(shè)為1
2)retain 引用計(jì)數(shù)器+1 返回了經(jīng)過(guò)+1以后的當(dāng)前實(shí)例對(duì)象
3)release 引用計(jì)數(shù)器-1歧譬,并不一定是釋放
4)retainCount 獲取引用計(jì)數(shù)器的值
5)dealloc 當(dāng)實(shí)例對(duì)象被銷毀之前,系統(tǒng)自動(dòng)調(diào)用宵统。
一定要調(diào)[super dealloc]
6)autorelease,該方法是將該對(duì)象內(nèi)存的管理放到autoreleasepool中贵白。
ARC(自動(dòng)管理內(nèi)存,Xcode 4.1/iOS5以后的特性)崩泡,需要注意的是他是編譯器特性禁荒,編譯器幫助我們插入那些內(nèi)存管理方法,同時(shí)他和其他語(yǔ)言的垃圾回收機(jī)制也有很大的不同
對(duì)比:
ARC中強(qiáng)引用的strong角撞,相當(dāng)于MRC中的retain
ARC中弱引用的weak呛伴,相當(dāng)于MRC中的assign
ARC中保存基本數(shù)據(jù)類型的assign,跟MRC中的assign一樣
PS:在ARC中谒所,保存弱引用不要用assign热康,assign是用來(lái)保存基本數(shù)據(jù)類型的,如果要保存弱引用劣领,需要使用weak