objective-c記憶體管理基礎

來源:互聯網
上載者:User

    對於我們.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的知識。

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.