天天看點

android記憶體洩漏導緻的問題,解決Android使用Handler造成記憶體洩露問題

一、什麼是記憶體洩露?

Java使用有向圖機制,通過GC自動檢查記憶體中的對象(什麼時候檢查由虛拟機決定),如果GC發現一個或一組對象為不可到達狀态,則将該對象從記憶體中回收。也就是說,一個對象不被任何引用所指向,則該對象會在被GC發現的時候被回收;另外,如果一組對象中隻包含互相的引用,而沒有來自它們外部的引用(例如有兩個對象A和B互相持有引用,但沒有任何外部對象持有指向A或B的引用),這仍然屬于不可到達,同樣會被GC回收。

Android中使用Handler造成記憶體洩露的原因

private Handler handler = new Handler()

{

public void handleMessage(android.os.Message msg)

{

if (msg.what == 1)

{

noteBookAdapter.notifyDataSetChanged();

}

}

};

上面是一段簡單的Handler的使用。當使用内部類(包括匿名類)來建立Handler的時候,Handler對象會隐式地持有一個外部類對象(通常是一個Activity)的引用(不然你怎麼可能通過Handler來操作Activity中的View?)。而Handler通常會伴随着一個耗時的背景線程(例如從網絡拉取圖檔)一起出現,這個背景線程在任務執行完畢(例如圖檔下載下傳完畢)之後,通過消息機制通知Handler,然後Handler把圖檔更新到界面。然而,如果使用者在網絡請求過程中關閉了Activity,正常情況下,Activity不再被使用,它就有可能在GC檢查時被回收掉,但由于這時線程尚未執行完,而該線程持有Handler的引用(不然它怎麼發消息給Handler?),這個Handler又持有Activity的引用,就導緻該Activity無法被回收(即記憶體洩露),直到網絡請求結束(例如圖檔下載下傳完畢)。另外,如果你執行了Handler的postDelayed()方法,該方法會将你的Handler裝入一個Message,并把這條Message推到MessageQueue中,那麼在你設定的delay到達之前,會有一條MessageQueue -> Message -> Handler -> Activity的鍊,導緻你的Activity被持有引用而無法被回收。

二、記憶體洩露的危害

記憶體洩露的危害就是會使虛拟機占用記憶體過高,導緻OOM(記憶體溢出),程式出錯。

對于Android應用來說,就是你的使用者打開一個Activity,使用完之後關閉它,記憶體洩露;又打開,又關閉,又洩露;幾次之後,程式占用記憶體超過系統限制,FC。

三、解決方案

使用Handler導緻記憶體洩露的解決方法

方法一:通過程式邏輯來進行保護。

1.在關閉Activity的時候停掉你的背景線程。線程停掉了,就相當于切斷了Handler和外部連接配接的線,Activity自然會在合适的時候被回收。

2.如果你的Handler是被delay的Message持有了引用,那麼使用相應的Handler的removeCallbacks()方法,把消息對象從消息隊列移除就行了。

方法二:将Handler聲明為靜态類。

PS:在Java 中,非靜态的内部類和匿名内部類都會隐式地持有其外部類的引用,靜态的内部類不會持有外部類的引用。

靜态類不持有外部類的對象,是以你的Activity可以随意被回收。由于Handler不再持有外部類對象的引用,導緻程式不允許你在Handler中操作Activity中的對象了。是以你需要在Handler中增加一個對Activity的弱引用(WeakReference)。

代碼如下:

static class MyHandler extends Handler

{

WeakReference mWeakReference;

public MyHandler(Activity activity)

{

mWeakReference=new WeakReference(activity);

}

@Override

public void handleMessage(Message msg)

{

final Activity activity=mWeakReference.get();

if(activity!=null)

{

if (msg.what == 1)

{

noteBookAdapter.notifyDataSetChanged();

}

}

}

}

PS:什麼是WeakReference?

WeakReference弱引用,與強引用(即我們常說的引用)相對,它的特點是,GC在回收時會忽略掉弱引用,即就算有弱引用指向某對象,但隻要該對象沒有被強引用指向(實際上多數時候還要求沒有軟引用,但此處軟引用的概念可以忽略),該對象就會在被GC檢查到時回收掉。對于上面的代碼,使用者在關閉Activity之後,就算背景線程還沒結束,但由于僅有一條來自Handler的弱引用指向Activity,是以GC仍然會在檢查的時候把Activity回收掉。這樣,記憶體洩露的問題就不會出現了。

四、總結

android中的很多記憶體洩露都是由于在Activity中使用了非靜态内部類導緻的,我們在使用非靜态内部類一定要格外注意,如果該靜态内部類的執行個體對象的生命周期大于外部對象,那麼就有可能導緻記憶體洩露,推薦使用上面介紹的靜态類和弱引用的方法解決這種問題。

以上所述是小編給大家介紹的Android使用Handler造成記憶體洩露問題及解決方法,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回複大家的。在此也非常感謝大家對腳本之家網站的支援!