OC內(nèi)存管理

內(nèi)存管理的方式

為什么要管理內(nèi)存



iOS應(yīng)用程序出現(xiàn)Crash(閃退),90%的原因是因為內(nèi)存問 題。  
在一個擁有數(shù)十個甚至是上百個類的的工程里 ,查找內(nèi)存問題極其困難,      
學會內(nèi)存管理,能幫我們減少出錯的機率。 
  • 內(nèi)存問題體現(xiàn)在兩個-----內(nèi)存溢出、野指針異常。

  • 內(nèi)部溢出

iOS給每個應(yīng)用程序分配了一定的內(nèi)存,  
用于程序的運行。一旦超出內(nèi)存上限,程序就會Crash。
  • 野指針異常
對象的內(nèi)存已經(jīng)被系統(tǒng)回收,但是仍然使用指針操作這塊內(nèi)存。  
野指針異常是程序Crash的重要原因之一。  
代碼量越大程序越容易出現(xiàn)野指針問題。

內(nèi)存管理方式

  • 垃圾回收機制(Garbage Collection)。

  • 垃圾回收機制:程序員只需要開辟內(nèi)存空間,不需要用代碼的形式釋放,系統(tǒng)來判斷哪些空間不再被使用,并回收這些內(nèi)存空間,以便再次分配。整個回收的過程不需要寫任何代碼,由系統(tǒng)自動完成垃圾回收。Java開發(fā)中一直使用的就是垃圾回收技術(shù)。

  • MRC(Manual Reference Counting)。
    -人工引用計數(shù):內(nèi)存的開辟和釋放 都由程序代碼進 控制。相對垃圾回收來說,對內(nèi)存的控制更 加靈活,可以在需要釋放的時候及時釋放,對程序員的要求較高,程序員要熟悉內(nèi)存管理的機制。
  • ARC(Auto Reference Counting)。

  • 自動引用計數(shù):iOS 5.0的編譯器特性,它允許用戶只開辟空間,不 去釋放空間。它不是垃圾回收!它的本質(zhì)還是MRC,只是編譯器幫程序員默認加了釋放的代碼。

  • ARC是基于MRC的

引用計數(shù)

  • C語言中,使用malloc和free,進行堆內(nèi)存的創(chuàng)建和釋放.堆內(nèi)存只有正在使用和銷毀兩種狀態(tài).
  • 實際開發(fā)中,可能會遇到兩個以上的指針使用同一塊內(nèi)存。C語言無法記錄內(nèi)存使用者的個數(shù).
OC對象的操作 OC中對應(yīng)的方法
生成對象 + alloc
持有對象 - retain
釋放對象 -release/-autorelease
銷毀對象 - dealloc
  • 影響引用計數(shù)的方法

    • +alloc: 開辟內(nèi)存空間,讓被開辟的內(nèi)存空間的引用計數(shù) 從0變?yōu)?。
    • -retain: 引用計數(shù)加1,如果對象之前引用計數(shù)為1,retain之后變?yōu)?
    • -copy: 把某一對象的內(nèi)容拷貝一份,拷貝出新的對象, 原有對象的引用計數(shù)不變,新的對象的引用計數(shù)變1。
    • -release: 引用計數(shù)立即減1,如果對象之前的引 計數(shù)為 ,release之后變?yōu)?,如果之前引用計數(shù)為1,release之后變?yōu)?,內(nèi)存被系統(tǒng)回收。
    • -autorelease: 未來的某一時刻引 計數(shù)減1 如果對象之前引用計數(shù)為4,autorelease之后仍然為4,未來某一刻會變?yōu)?。

autoreleasepool的使用

  • 通過autoreleasepool自動釋放池,控制autoreleasepool對象的釋放。
  • 向一個對象發(fā)送autorelease消息,該對象就會被添加到離autorelease最近的自動釋放池中,當自動釋放池銷毀時,為池中的每一個對象發(fā)送release消息。

NSAutoreleasePool


iOS5之前,使用NSAutoreleasePool自動釋放池類創(chuàng)建對象。  

NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init]; // 動釋放池創(chuàng)建
[pool release]; // 動釋放池銷毀

dealloc

  • -dealloc是繼承父類的方法,當對象引用計數(shù)為0的時候,由對象自動調(diào)用 ,銷毀該對象的空間。
  • 重寫dealloc方法,驗證對象的空間是否被回收。
- (void)dealloc {
    
    //這里面寫一些本類使用的其他資源的銷毀操作,并且需要在父類dealloc方法調(diào)用之前
    NSLog(@"調(diào)用了銷毀");
    //MRC下重寫dealloc方法需要調(diào)用父類的實現(xiàn),ARC下重寫dealloc方法不需要寫[super dealloc],系統(tǒng)會自動調(diào)用
    [super dealloc];
}

  • 下面代碼會出現(xiàn)什么樣的問題
    @autoreleasepool {
        for (NSInteger i = 0; i < 1000000000000; i++) {
            Person *p = [[Person alloc]init];
            [p autorelease];
        }
    }
//因為autorelease 出自動釋放池才釋放  所以造成內(nèi)存增加
  • 解決方法

    
    @autoreleasepool {
        for (NSInteger i = 0; i < 1000000000000; i++) {
            @autoreleasepool {
                Person *p = [[Person alloc]init];
                [p autorelease];
            }
        }
    }

    //把一個對象添加到一個集合中會對該對象的引用計數(shù)加1,如果移除會將該對象的引用計數(shù)減1
    Person *p4 = [[Person alloc] init];
    NSMutableArray *arr = [NSMutableArray array];
    
    [arr addObject:p4];
    [p4 release];//release alloc的+1

內(nèi)存管理原則

凡是使用了alloc,retain,copy讓內(nèi)存的引用計數(shù) 增加了,就需要使release或者autorele讓內(nèi)存的引 計數(shù)減少。在一段代碼內(nèi),增加和減少的次數(shù)要相等。
   如果增加的次數(shù)多于減少的次數(shù),會造成內(nèi)存泄露。
   如果增加次數(shù)少于減少的次數(shù),會造成過度釋放。
   如果增加的次數(shù)等于減少的次數(shù),還繼續(xù)訪問,造成野指針問題。
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • OC內(nèi)存管理一、基本原理(一)為什么要進行內(nèi)存管理。由于移動設(shè)備的內(nèi)存極其有限,所以每個APP所占的內(nèi)存也是有限制...
    ScaryMonsterLyn閱讀 582評論 0 3
  • 今天看到一篇不錯的文章關(guān)于OC內(nèi)存管理的,轉(zhuǎn)載一下與你共享概述我們知道在程序運行過程中要創(chuàng)建大量的對象,和其他高級...
    niceSYT閱讀 514評論 0 2
  • OC內(nèi)存管理 一、基本原理 (一)為什么要進行內(nèi)存管理。 由于移動設(shè)備的內(nèi)存極其有限,所以每個APP所占的內(nèi)存也是...
    iOS_Developer閱讀 429評論 0 3
  • Objective-C提供三種內(nèi)存管理模型: 自動垃圾回收 手動引用計數(shù)MRC和自動釋放池. 自動引用計數(shù)ARC....
    Fsn_soul閱讀 323評論 0 0
  • 內(nèi)存管理 簡述OC中內(nèi)存管理機制。與retain配對使用的方法是dealloc還是release,為什么?需要與a...
    丶逐漸閱讀 2,081評論 1 16

友情鏈接更多精彩內(nèi)容