詳解Android使用Handler造成內(nèi)存泄露的分析及解決方法
一、什么是內(nèi)存泄露?
Java使用有向圖機制,通過GC自動檢查內(nèi)存中的對象(什么時候檢查由虛擬機決定),如果GC發(fā)現(xiàn)一個或一組對象為不可到達狀態(tài),則將該對象從內(nèi)存中回收。也就是說,一個對象不被任何引用所指向,則該對象會在被GC發(fā)現(xiàn)的時候被回收;另外,如果一組對象中只包含互相的引用,而沒有來自它們外部的引用(例如有兩個對象A和B互相持有引用,但沒有任何外部對象持有指向A或B的引用),這仍然屬于不可到達,同樣會被GC回收。
Android中使用Handler造成內(nèi)存泄露的原因
private Handler handler = new Handler(){
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
}
};
我們經(jīng)常使用handler時,會像如上那樣定義,但是AS的lint檢測會自動報錯警告提示,如下圖

上面是一段簡單的Handler的使用。當使用內(nèi)部類(包括匿名類)來創(chuàng)建Handler的時候,Handler對象會隱式地持有一個外部類對象(通常是一個Activity)的引用(不然你怎么可能通過Handler來操作Activity中的View?)。而Handler通常會伴隨著一個耗時的后臺線程(例如從網(wǎng)絡(luò)拉取圖片)一起出現(xiàn),這個后臺線程在任務(wù)執(zhí)行完畢(例如圖片下載完畢)之后,通過消息機制通知Handler,然后Handler把圖片更新到界面。然而,如果用戶在網(wǎng)絡(luò)請求過程中關(guān)閉了Activity,正常情況下,Activity不再被使用,它就有可能在GC檢查時被回收掉,但由于這時線程尚未執(zhí)行完,而該線程持有Handler的引用(不然它怎么發(fā)消息給Handler?),這個Handler又持有Activity的引用,就導致該Activity無法被回收(即內(nèi)存泄露),直到網(wǎng)絡(luò)請求結(jié)束(例如圖片下載完畢)。另外,如果你執(zhí)行了Handler的postDelayed()方法,該方法會將你的Handler裝入一個Message,并把這條Message推到MessageQueue中,那么在你設(shè)定的delay到達之前,會有一條MessageQueue -> Message -> Handler -> Activity的鏈,導致你的Activity被持有引用而無法被回收。
二、內(nèi)存泄露的危害
內(nèi)存泄露的危害就是會使虛擬機占用內(nèi)存過高,導致OOM(內(nèi)存溢出),程序出錯。
對于Android應(yīng)用來說,就是你的用戶打開一個Activity,使用完之后關(guān)閉它,內(nèi)存泄露;又打開,又關(guān)閉,又泄露;幾次之后,程序占用內(nèi)存超過系統(tǒng)限制,F(xiàn)C。
三、解決方案
使用Handler導致內(nèi)存泄露的解決方法
方法一(官方解決辦法):
private Handler mHandler2 = new Handler(new Handler.Callback() {
@Override
public boolean handleMessage(Message msg) {
//do something
return false;
}
});
方法二:通過程序邏輯來進行保護。
1.在關(guān)閉Activity的時候停掉你的后臺線程。線程停掉了,就相當于切斷了Handler和外部連接的線,Activity自然會在合適的時候被回收。
2.如果你的Handler是被delay的Message持有了引用,那么使用相應(yīng)的Handler的removeCallbacks()方法,把消息對象從消息隊列移除就行了。
方法三:將Handler聲明為靜態(tài)類。
PS:在Java 中,非靜態(tài)的內(nèi)部類和匿名內(nèi)部類都會隱式地持有其外部類的引用,靜態(tài)的內(nèi)部類不會持有外部類的引用。
靜態(tài)類不持有外部類的對象,所以你的Activity可以隨意被回收。由于Handler不再持有外部類對象的引用,導致程序不允許你在Handler中操作Activity中的對象了。所以你需要在Handler中增加一個對Activity的弱引用(WeakReference)。
代碼如下:
static class MyHandler extends Handler {
WeakReference<Activity> mWeakReference;
public MyHandler(Activity activity) {
mWeakReference = new WeakReference<Activity>(activity);
}
@Override
public void handleMessage(Message msg) {
final Activity activity = mWeakReference.get();
if (activity != null) {
if (msg.what == 1) {
activity.noteBookAdapter.notifyDataSetChanged();
}
}
}
}
不管是任何方法,再當前界面聲明周期結(jié)束之后,記得移除handler隊列中的消息
handler.removeCallbacksAndMessages(null);
PS:什么是WeakReference?
WeakReference弱引用,與強引用(即我們常說的引用)相對,它的特點是,GC在回收時會忽略掉弱引用,即就算有弱引用指向某對象,但只要該對象沒有被強引用指向(實際上多數(shù)時候還要求沒有軟引用,但此處軟引用的概念可以忽略),該對象就會在被GC檢查到時回收掉。對于上面的代碼,用戶在關(guān)閉Activity之后,就算后臺線程還沒結(jié)束,但由于僅有一條來自Handler的弱引用指向Activity,所以GC仍然會在檢查的時候把Activity回收掉。這樣,內(nèi)存泄露的問題就不會出現(xiàn)了。
四、總結(jié)
android中的很多內(nèi)存泄露都是由于在Activity中使用了非靜態(tài)內(nèi)部類導致的,我們在使用非靜態(tài)內(nèi)部類一定要格外注意,如果該靜態(tài)內(nèi)部類的實例對象的生命周期大于外部對象,那么就有可能導致內(nèi)存泄露,推薦使用上面介紹的靜態(tài)類和弱引用的方法解決這種問題。
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
- Android開發(fā)中Google為什么不讓用Handler的runWithScissors()
- Android Handler,Message,MessageQueue,Loper源碼解析詳解
- Android Handle原理(Looper,Handler和Message)三者關(guān)系案例詳解
- Android Handler消息機制分析
- 詳解Android Handler的使用
- android利用handler實現(xiàn)倒計時功能
- android利用handler實現(xiàn)打地鼠游戲
- Android Handler實現(xiàn)閃屏頁倒計時代碼
- 使用C++11實現(xiàn)Android系統(tǒng)的Handler機制
- Android Handler機制的工作原理詳析
- 深入Android HandlerThread 使用及其源碼完全解析
- 掌握Android Handler消息機制核心代碼
相關(guān)文章
淺談Android系統(tǒng)的基本體系結(jié)構(gòu)與內(nèi)存管理優(yōu)化
這篇文章主要介紹了Android系統(tǒng)的基本體系結(jié)構(gòu)與內(nèi)存管理優(yōu)化,非常簡潔明了地總結(jié)了系統(tǒng)服務(wù)及垃圾回收等安卓的一些主要特性,需要的朋友可以參考下2016-02-02
Android?Jetpack組件Navigation導航組件的基本使用
本篇主要簡單介紹了一下?Navigation?是什么?以及使用它的流程是什么,并且結(jié)合實際案例?操作了一番,Navigation?還有很多其他用法,如條件導航、嵌套圖、過度動畫?等等功能?有機會再操作,需要的朋友可以參考下2022-06-06
Android基于IJKPlayer視頻播放器簡單封裝設(shè)計
這篇文章主要介紹了Android基于IJKPlayer視頻播放器簡單封裝設(shè)計,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-06-06
Android使用ViewFlipper實現(xiàn)圖片切換功能
這篇文章主要為大家詳細介紹了Android使用ViewFlipper實現(xiàn)圖片切換功能,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-07-07
Android自定義View實現(xiàn)搜索框(SearchView)功能
這篇文章主要為大家詳細介紹了Android自定義View實現(xiàn)搜索框SearchView功能,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-11-11
Android MediaPlayer 音頻倍速播放 調(diào)整播放速度問題
這篇文章主要介紹了Android MediaPlayer 音頻倍速播放,調(diào)整播放速度,本文給大家介紹的非常詳細,具有一定的參考借鑒價值,需要的朋友可以參考下2019-09-09
Android下的CMD命令之關(guān)機重啟及重啟recovery
這篇文章主要介紹了Android下的CMD命令之關(guān)機重啟及重啟recovery,本文涉及到cmd命令知識點,通過了解cmd命令就可以很容易的實現(xiàn)此功能了,需要的朋友一起看看吧2016-08-08

