對於我們.net開發人員來說,.net為我們提供了自動記憶體管理的機制,我們不需去關心記憶體的管理。但是iphone開發中卻是不能的。這篇文章將簡述一下objective-c的記憶體管理機制和方法和一些特性。
手動的進行記憶體管理
Cocoa和Objective-C的類都是NSObject的子類。NSObject中有幾個方法進行記憶體管理。alloc方法為對象分配一片記憶體空間。dealloc方法用於釋放對象的空間。但是在我們的代碼中將永遠都不會使用dealloc方法,因為運行時會為你調用此方法釋放記憶體空間。而你需要做的只是引用計數,稍後介紹什麼是引用計數。
除了alloc和dealloc,NSObject的還有retain和release方法兩個方法用於引用計數。retain方法給retainCount變數加1,release方法給retainCount變數減1。當使用alloc為對象分配一片記憶體空間的時候,retainCount會為1。在這個對象的生命週期內,這個對象可能繼續被其它變數引用。但有新的變數指向這個對象的時候,你應該調用retain方法,這樣運行時才會知道有新的引用指向了這個變數,在這個物件存留期中擁有它的使用權。這個被Objective-C開發人員稱之為“擁有”。例如:
Foo * myFooOne = [[Foo alloc] init]; //retaincount 為1
Foo * myFooTwo = myFooOne; //myFooTwo 指向了這個對象
//retaincount 仍然為1
[myFooTwo retain]; //調用retain方法,運行時才知道myFooTwo指向了該對象,retaincount 為2
上面的代碼中,myFooTwo通過調用retain方法,取得了Foo對象的擁有權。在這個對象的生命週期中,會有很多變數來指向和引用它。指向這個對象的變數也可以通過release方法來解除這種擁有權。release方法將會告訴運行時,我已經使用完這個變數了,已經不需要它了,retainCount計數減1。
當對象的retainCount的計數大於或者等於1的時候,運行時會繼續維持這個對象。當對象的retainCount為0的時候,運行時會釋放這個對象,並回收它佔得記憶體空間。
展示了一個Foo對象的生命週期。Foo對象首先在記憶體中分配一個記憶體空間,並且被myFooOne引用。在這個時候Foo對象的retaincount為1。
Foo * myFooOne = [[Foo alloc] init];
第二個引用變數指向Foo對象,這個引用變數接著調用retain方法,其實也是調用Foo對象的retain方法。Foo對象的retaincount變成2。
Foo * myFooTwo = myFooOne;
[myFooTwo retain];
接著當myFooOne引用不需要的時候,通過調用release方法,解除與Foo對象的擁有權,Foo對象的retaincount變成1。
[myFooOne release];
但myFooTwo不在需要的時候,同樣通過調用release方法,解除與Foo對象的擁有權,Foo對象的retaincount變成0。
記憶體泄露
我們經常會在一個方法中聲明對象,看下面這個例子:
-(void) myMethod {
//incorrect method
NSString * myString = [[NSString alloc] init]; //retainCount = 1
Foo * myFoo = [[Foo alloc] initWithName:myString]; //retainCount = 1
NSLog(@"Foo's Name:%@", [myFoo getName]);
}
這上面這個方法中,我們為myString 和myFoo分配了記憶體空間。方法執行結束之後,兩個變數超出了範圍的範圍,所以不再有效。但是這個方法並沒有releases這兩個對象。所以運行時沒有釋放這兩個變數佔據的記憶體空間。除非你的應用程式結束,否則這兩個變數佔據的記憶體空間一直都是停用。我們把它稱之為記憶體泄露。
為了防止記憶體泄露。無論什麼時候,我們建立一個對象,或者建立一個對象的拷貝,我們都必須通過release方法釋放。
-(void) myMethod {
NSString * myString = [[NSString alloc] init]; //retainCount=1
Foo * myFoo = [[Foo alloc] initWithName:myString]; //retainCount=1
NSLog("Foo's Name:%@", [myFoo getName]);
[myFoo release]; //retainCount=0 so deallocate
[myString release]; //retainCount=0 so deallocate
}
弱引用
看下面的例子:
-(void) myMethod {
//an incorrect method
Foo * myFooOne = [[Foo alloc] initWithName:@"James"]; //retainCount=1
Foo * myFooTwo = myFooOne; //retainCount still 1
[myFooOne release]; //retaincount=0 so deallocated
NSLog("Name:%@", [myFooTwo printOutName]); //runtime error
}
nyFooTwo指向了Foo對象,但是沒有調用retain方法,就是一種弱引用,上面的代碼會在運行時報錯。因為myFooOne調用release方法。retaincount變成0,運行時,回收了對象的記憶體空間。然後myFooTwo調用printPutName自然就報錯了,見說明。
總結:本文簡單的介紹了一下手動的進行記憶體管理、記憶體泄露、弱引用等objective-c的知識。