首頁 > 軟體

Android記憶體漏失導致原因深入探究

2023-02-20 06:00:25

什麼是記憶體洩露

什麼是記憶體洩露,通俗的來說就是堆中的一些物件已經不會再被使用了,但垃圾收集器卻無法將它們從記憶體中清除。

記憶體漏失很嚴重的問題,因為它會阻塞記憶體資源並隨著時間的推移降低系統效能。如果不進行有效的處理,最終的結果將會使應用程式耗盡記憶體資源,無法正常服務,導致程式崩潰,丟擲java.lang.OutOfMemoryError異常。

堆記憶體中通常有兩種型別的物件:被參照的物件和未被參照的物件。被參照的物件是應用程式中仍然具有活躍的參照,而未被參照的物件則沒有任何活躍的參照。

垃圾收集器會回收那些未被參照的物件,但不會回收那些還在被參照的物件。這也是記憶體洩露發生的源頭。

哪些操作會造成記憶體漏失

下面我們介紹幾種常見的造成記憶體洩露的情況

1、意外宣告全域性變數是最常見也最容易修復的記憶體漏失問題,比如:

function fn() {
    name = '張三';
}

直譯器在解釋上面的函數時,會把name當做全域性變數,即window.name = ‘張三’。只要window物件沒有被清理,那麼name屬性和屬性值將一直存在,造成記憶體洩露。

解決方法:

(1)只要在變數宣告前面加上var、let或const關鍵字即可,這樣變數就會在函數執行完畢後離開作用域。

(2)使用this關鍵字

function fn() {
    this.name = '張三';
}

(3)可以在 JavaScript 檔案開頭新增 “use strict”,使用嚴格模式。這樣在嚴格模式下解析 JavaScript 可以防止意外的全域性變數

(4)在使用完之後,對其賦值為null或者重新分配

2、 定時器導致的洩露

let name = '張三';
setInterval(() => {
    console.log(name);
}, 100);

上面的程式碼中,只要定時器一直執行,回撥函數中參照的name就會一直佔用記憶體。

3、閉包、控制檯紀錄檔、迴圈(在兩個物件彼此參照且彼此保留時,就會產生一個迴圈),下面我們看一個JavaScript閉包導致的內訓洩露例子

let fun = function() {
    let name = '張三';
    return function() {
        return name;
    };
};

呼叫fun()會導致分配給name的記憶體被洩漏。以上程式碼執行後建立了一個內部閉包,只要返回的函數存在就不能清理name,因為閉包一直在參照著它。

常見記憶體洩露問題

1.資源性物件未關閉

資源性物件(如Cursor、File等一些Closeable物件),它們往往使用了緩衝區,緩衝區不僅在JVM內,JVM之外也有。如果僅僅把變數設定為null,而不關閉它們,緩衝區得不到釋放,往往造成記憶體洩露。

解決方案:一般在finally中關閉資源型物件,而後設定物件為null

2.註冊物件未登出

訂閱者模式中,如果註冊物件不再使用時,未及時登出,會導致訂閱者列表中維持這物件的參照,阻止垃圾回收,導致記憶體洩露。常見場景:動態註冊BroadcastReceiver,註冊PhoneStateListener,註冊EventBus等等,

還有自定義使用訂閱者模式的情形。

解決方案:一般在onDestroy()中進行解註冊

3.非靜態內部類的靜態範例

非靜態內部類持有外部類範例的參照,若非靜態內部類的範例是靜態的,便擁有app存活期整個生命週期,長期持有外部類的參照,阻止外部類範例被回收。

使用內部類的情況十分常見,尤其是匿名內部類:一些介面的匿名實現類,都是內部類。

解決方案:(1)改為靜態內部類,不再持有外部類範例的參照 (2)避免申明非靜態內部類的靜態範例 (3)將內部類抽取出來封裝成一個單例,如果需要Context,沒有特殊要求就使用Application Context;如果需要Activity Context,則使用完畢置空,或者使用弱參照

4.單例模式引起的記憶體洩露

由於單例模式的靜態特性,使得它的生命週期和我們的應用一樣長,如果讓單例無限制的持有Activity的強參照就會導致記憶體漏失

解決方案:使用Activity的弱參照,或者沒特殊需求時使用Application Context

5.Handler臨時性記憶體洩露

非靜態Handler持有Activity或Service的參照,Message中的target指向Handler範例,所以當Message在MessageQueue中排隊,長時間未得到處理時,Activity邊不會被回收,導致臨時性記憶體洩露。

解決方案:(1)使用靜態Handler內部類,然後對Handler持有的物件(Activity或Service)使用弱參照 (2)在onDestroy()中移除訊息佇列中的訊息 mHandler.removeCallbacksAndMessages(null)

類似的:AsyncTask內部也是Handler機制,也存在同樣的臨時性記憶體洩露風險

6.容器中物件未及時清理導致記憶體洩露

容器類一般擁有較長的生命週期,若內部不再使用的物件不及時清理,內部物件邊一直被容器類參照。上述2中的訂閱者列表也屬於容器類這中情況。另外常見的容器類還有執行緒池、物件池、圖片快取池等。執行緒池中的執行緒若存在ThreadLocal物件,因為執行緒物件一直被迴圈使用,ThreadLocal物件便會一直被參照,要注意對value物件的置空釋放。

7.靜態View導致記憶體洩露

有時,當一個Activity經常啟動,但是對應的View讀取非常耗時,我們可以通過靜態View變數來保持對該Activity的rootView參照。這樣就可以不用每次啟動Activity都去讀取並渲染View了。這確實是一個提高Activity啟動速度的好方法!但是要注意,一旦View attach到我們的Window上,就會持有一個Context(即Activity)的參照。而我們的View有事一個靜態變數,所以導致Activity不被回收。 解決辦法:在使用靜態View時,需要確保在資源回收時,將靜態View detach掉。

8.屬性動畫未及時關閉導致記憶體洩露

在使用ValueAnimator或者ObjectAnimator時,如果沒有及時做cancel取消動畫,就可能造成記憶體洩露。 因為在cancel方法裡,最後呼叫了endAnimation(); ,在endAnimation裡,有個AnimationHandler的單例,會持有屬性動畫物件的參照

解決辦法:在在onDestory時,呼叫動畫的cancel方法

9.WebView記憶體洩露

目前Android中WebView的實現存在很大的相容性問題,Google支援各個ROM廠商自行客製化自己的WebView實現,各個ROM間差異較大,且大多都存在記憶體洩露問題。除了呼叫其內部的clearCache()、clearHistory()、removeAllViews()、freeMemory()、destroy()和置null以外,一般比較粗暴有效的解決方法是:將包含WebView的Activity放在一個單獨的程序中,不需要時將程序銷燬,從而釋放所有所佔記憶體。

10.其他的系統控制元件以及自定義View

在 Android Lollipop 之前使用 AlertDialog 可能會導致記憶體漏失

view中有執行緒或者動畫 要及時停止。這是為了防止記憶體漏失,可以在onDetachedFromWindow方法中結束,這個方法回撥的時機是 當View的Activity退出或者當前View被移除的時候 會呼叫 這時候是結束動畫或者執行緒的好時機 另外還有一個對應的方法 onAttachedToWindow 這個方法呼叫的時機是在包含View的Activity啟動時 回撥 回撥在onDraw方法之前

11.其他常見的引起記憶體漏失原因

  • (1)構造Adapter時,沒有使用快取的 contentView
  • (2)Bitmap在不使用的時候沒有使用recycle()釋放記憶體
  • (3)警惕執行緒未終止造成的記憶體洩露;譬如在Activity中關聯了一個生命週期超過Activity的Thread,在退出Activity時切記結束執行緒。一個典型的例子就是HandlerThread的run方法是一個死迴圈,它不會自己結束,執行緒的生命週期超過了Activity生命週期,我們必須手動在Activity的銷燬方法中呼叫thread.getLooper().quit();才不會洩露
  • (4)避免程式碼設計模式的錯誤造成記憶體洩露;譬如迴圈參照,A持有B,B持有C,C持有A,這樣的設計誰都得不到釋放

文末

理解記憶體漏失的危害,我們舉個簡單的例子。有一個賓館,有100間房間,顧客每次都是在前臺進行登記,然後拿到房間鑰匙。如果有些顧客不需要該房間了,也不歸還鑰匙,久而久之,前臺處可用房間越來越少,收入也越來越少,瀕臨倒閉。當程式申請了記憶體,而不進行歸還,久而久之,可用記憶體越來越少,OS就會進行自我保護,殺掉該程序,這就是我們常說的OOM(out of memory)。

到此這篇關於Android記憶體漏失導致原因深入探究的文章就介紹到這了,更多相關Android記憶體漏失內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com