Objective-C記憶體管理之-引用計數

来源:http://www.cnblogs.com/denz/archive/2016/03/30/5339038.html
-Advertisement-
Play Games

本文會繼續深入學習OC記憶體管理,內容主要參考iOS高級編程,Objective C基礎教程,瘋狂iOS講義,是我學習記憶體管理的筆記 記憶體管理 1 記憶體管理的基本概念 1.1 Objective C中的記憶體管理 手動記憶體管理和自動釋放池 \ (Mannul Reference Counting) 自動 ...


本文會繼續深入學習OC記憶體管理,內容主要參考iOS高級編程,Objective-C基礎教程,瘋狂iOS講義,是我學習記憶體管理的筆記

記憶體管理

1 記憶體管理的基本概念

1.1 Objective-C中的記憶體管理

  • 手動記憶體管理和自動釋放池---MRC>(Mannul Reference Counting)
  • 自動記憶體管理---ARC>(Automatic Reference Count)
  • 自動垃圾回收---GC>(Garbage Collection)
由於iOS系統不支持垃圾回收,所以我們在iOS開發中只能使用MRC和ARC來進行記憶體管理,本文不再介紹Objective-C中的垃圾回收機制,但是此處註意Objective-C中是存在垃圾回收機制的

1.2 記憶體管理中存在的問題

  • 記憶體泄露:不再需要的對象沒有釋放

引起的問題:程式的記憶體占有量不斷增加,最終會被耗盡導致程式崩潰

  • 野指針:沒有進行初始化得指針

引起的問題:浪費記憶體資源,如果調用程式會出現未知的結果,甚至導致程式崩潰

  • 懸空指針:一個指針指向一個被銷毀的對象

引起的問題:調用懸空指針指向的屬性或者方法時,程式會出現未知的結果,甚至導致程式崩潰

  • 僵屍對象:過度釋放的對象

引起的問題:

2.手動記憶體管理和自動釋放池---MRC>(Mannul Reference Counting)

2.1 什麼是引用計數(Reference Counting)

引用計數:Objective-C中引入了引用計數這一機制來跟蹤並處理對象的生命周期,

管理方式:每個對象都有一個與之關聯的整數,這個整數被稱為引用計數,在Objective-C中,通過不同的方法可以對引用計數進行操作,具體的處理如下表:

對象操作 Objective-C方法 對應的操作結果
生成並持有對象 alloc, new, copy,mutableCopy等方法 生成對象並設置引用計數 =1
持有對象 reatain方法 使引用計數 +1
釋放對象 release方法 使引用計數 -1
廢棄對象 dealloc方法---系統自動調用 引用計數 =0 時調用

關於delloc方法:dealloc方法繼承自NSObject,因此所有的對象都具有此方法,當一個對象的引用計數為0時,也就意味著沒有任何程式需要此對象,系統會回收該對象所占用的記憶體,在系統銷毀對象之前,會自動調用該對象的dealloc方法來執行一些回收操作,如果該對象還持有其他對象的引用,我們必須重寫dealloc方法來釋放該對象引用的其他對象(通常就是使用該對象的release方法)

引用計數機制回收對象的說明:如果一個對象的引用計數為0,則表明程式已經不再需要它,這時系統會自動回收該對象所占記憶體,相反,如果一個對象的引用計數不為0,系統就不應該回收,也不會回收它所占的記憶體

關於retainCount方法:Objective-C提供了retainCount方法來返回一個對象當前的引用計數

如何重寫dealloc方法:

- (void)dealloc {

    // 處理該對象的其他引用(通過release方法)
    
    /** 回調父類的dealloc方法 */
    [super dealloc];
}

2.2 蘋果如何管理引用計數

  • 2.2.1 因為NSObject類的源代碼沒有公開,我們利用Xcode的調試器(lldb)和iOS大概追溯出其實現過程
    • alloc

      +alloc
      +allocWithZone: 
      class_createInstance        //此方法可以通過objc4中的runtime/objc-runtime-new.mm確認
      calloc                      // 分配記憶體塊
    • retainCount

      -retainCount 
      __CFDoExternRefOperation    // 此函數根據retain,retainCount,release操作進行分發,調用__CFBasicHashXXX方法
      CFBasicHashGetCountOfKey
    • retain

      -retain
      __CFDoExternRefOperation 
      CFBasicHashAddValue
    • release

      -release 
      __CFDoExternRefOperation 
      CFBasicHashRemoveValue      // 當此函數返回0時, -release調用dealloc方法
  • 2.2.2 由__CFDoExternRefOperation函數以及此函數的調用關係,我們大概推算蘋果大概是使用散列表(引用計數表)來管理引用計數
    • 通過引用計數表來管理引用計數的好處:
      • 對象用記憶體塊的分配無須考慮記憶體塊頭部
      • 引用計數表各記錄中存有記憶體塊的地址,可從各個記錄追溯到各對象的記憶體塊(在進行記憶體泄露的檢查時,此條特性具有舉足輕重的作用,即使出現故障導致對象占用的記憶體塊損壞,但是只要引用計數表沒有被破壞,我們就可以確定各記憶體塊的位置,這就是設置全局斷點可以查出哪裡出現記憶體泄露的原因)
        引用計數表

2.3 記憶體管理的思考方式

  1. 自己生成的對象,自己持有

    1.1 使用alloc new copy mutableCopy創建的對象只能自己持有

    id obj1 = [[NSObject alloc] init];
    id obj2 = [NSObject new];
    id obj3 = [NSObject copy];
    id obj4 = [NSObject mutableCopy];

    1.2 使用以上名稱的開頭的方法也意味著自己生成並持有對象

    allocNewObject

    newNewObject

    copyNewObject

    mutableCopyNewObject

  2. 非自己生成的對象,自己也能持有

    2.1 非alloc new copy mutableCopy生成的對象,變數obj本身不持有該對象

    id obj1 = [NSMutableArray array];
    id obj2 = [NSDictionary dictionary];
    

    2.2 通過retain方法,非通過alloc new copy mutableCopy生成的對象,可以成為自己持有的對象

    id obj = [NSMutableArray array];
    
    [obj retain];
  3. 不再需要自己持有的對象時釋放

    • 3.1 釋放通過alloc new copy mutableCopy生成的對象,一旦不在需要,務必要使用release方法釋放

    ```
    id obj = [[NSObject alloc] init];

    [obj release];
    ```

    • 3.2 用retain方法持有的非自己生成的對象,一旦不再需要,也一定要使用release釋放

    id obj = [NSMutableArray array]; [obj retain]; // 通過retain方法持有對象 [obj release]; // 在不需要時也要通過release方法釋放對象

    • 3.3 用某個方法生成對象,並將其作為方法的返回值,這時我們該如何處理

      • 3.3.1 通過alloc new copy mutableCopy 或其他符合命名規則的方法生成的對象,只需要原封不動的返回就能讓調用方也持有該對象

        ```
        • (id)allocObject {
          id obj = [[NSObject alloc] init];
          return obj;
          }

        • (id)allocObjectWithObject:(id)obj {
          id object = [obj allocObject];
          return object;
          }
          ```

      • 3.3.2 如果持有非自己生成的對象,例如[NSMutableArray array]生成的對象,我們要使用autorelease方法釋放

        註:命名規則:用來取得誰都不持有的對象的方法名不能以alloc new copy mutableCopy開頭

        - (id)object {  
             id obj = [NSMutableArray array];
             [obj autorelease];
             return obj;
        }
      • 3.3.3 autorelease方法:提供了這樣的功能,使對象在超出指定的生存範圍時自動並正確釋放(調用release方法)
        release和autorelease的區別

  4. 非自己持有的對象無法釋放---註意以下兩點,如果發生這樣的情況會導致程式崩潰

  • 4.1 通過alloc new copy mutableCopy方法或者通過retain方法持有的對象,一旦不再需要時,必須進行釋放,除此之外其他方法獲得的對象絕對不能釋放,一旦釋放會造成程式崩潰

  • 4.2 自己持有的對象釋放後再次釋放,造成僵死對象,引起程式崩潰或在訪問廢棄的對象時崩潰

    id obj = [[NSObject alloc] init]; [obj release]; [obj release]; // 再次釋放


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 通過使用該源碼,開發者可以迅速地將Discuz論壇遷移到Android客戶端中。不需要任何的開發工作即可擁有屬於自己論壇的Android客戶端 源碼下載:http://code.662p.com/view/13266.html 準備工作 在使用源碼之前必須先在Discuz論壇中安裝BigApp插件。 ...
  • Android 開發了一段時間,一方面 ,感覺不留下點什麼。有點對不起自己, 另一方面,好記性不如爛筆頭,為了往後可以回頭來看看,就當做是筆記,便決定開始寫博客。廢話不多說 ! 今天想搞一搞 ndk 和jni ,, 現在開始寫一個簡單的demo 1. 創建一個新的工程 2. 創建一個新的類 JniT ...
  • iOS App的性能關註點 雖然iPhone的機能越來越好,但是app的功能也越來越複雜,性能從來都是移動開發的核心關註點之一。我們說一個app性能好,不是簡單指感覺運行速度快,而應該是指應用啟動快速、UI反饋響應及時、列表滾動操作流暢、記憶體使用合理,當然更不能隨隨便便Crash啦。工程師開發應用時 ...
  • 本文內容 環境 android-common 項目結構 演示 android-common 參考資料 android-common 主要包括如下內容: 緩存,包括圖片緩存、預取緩存、網路緩存。 公共 View,即功能封裝好的部件,包括下拉獲得最新和上拉載入更多 ListView、底部載入更多 Scr ...
  • 匿名內部類作為事件監聽器類實現頁面跳轉 @Overrideprotected void onCreate(Bundle savedInstanceState){ super.onCreate(savedInstanceState); setContentView(R.layout.main1);/* ...
  • ```objc1 使用Crearte函數創建的併發隊列和全局併發隊列的主要區別: 1)全局併發隊列在整個應用程式中本身是預設存在的並且對應有高優先順序、預設優先順序、低優先順序和後臺優先順序一共四個併發隊列,我們只是選擇其中的一個直接拿來用。而Create函數是實打實的從頭開始去創建一個隊列。 2)在iOS ...
  • 效果如圖 1 首先這是一個自定義的Dialog,而不是AlertDialog,如果是AlertDialog的話,軟鍵盤彈出的時候在AlertDialog的後面,無法進行輸入。 2 Dialog的上面會有一個黑框,添加Style 3 輸入密碼用的是EditText,改變EditText的格式是 4 給 ...
  • 提到定時器,NStimer肯定是我們最為熟悉的。 但是NStimer有著很大的缺點,並不准確。 通俗點說,就是它該做他的事了,但是由於其他事件的影響,Nstimer會放棄他應該做的。 而GCD定時器,是不會發生這種事情的。 GCD嚴格按照規定好的規格去做事。 前面介紹RunLoop 的時候已經介紹了 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...