解決Android使用Handler造成內(nèi)存泄露問(wèn)題
一、什么是內(nèi)存泄露?
Java使用有向圖機(jī)制,通過(guò)GC自動(dòng)檢查內(nèi)存中的對(duì)象(什么時(shí)候檢查由虛擬機(jī)決定),如果GC發(fā)現(xiàn)一個(gè)或一組對(duì)象為不可到達(dá)狀態(tài),則將該對(duì)象從內(nèi)存中回收。也就是說(shuō),一個(gè)對(duì)象不被任何引用所指向,則該對(duì)象會(huì)在被GC發(fā)現(xiàn)的時(shí)候被回收;另外,如果一組對(duì)象中只包含互相的引用,而沒(méi)有來(lái)自它們外部的引用(例如有兩個(gè)對(duì)象A和B互相持有引用,但沒(méi)有任何外部對(duì)象持有指向A或B的引用),這仍然屬于不可到達(dá),同樣會(huì)被GC回收。
Android中使用Handler造成內(nèi)存泄露的原因
private Handler handler = new Handler() { public void handleMessage(android.os.Message msg) { if (msg.what == 1) { noteBookAdapter.notifyDataSetChanged(); } } };
上面是一段簡(jiǎn)單的Handler的使用。當(dāng)使用內(nèi)部類(包括匿名類)來(lái)創(chuàng)建Handler的時(shí)候,Handler對(duì)象會(huì)隱式地持有一個(gè)外部類對(duì)象(通常是一個(gè)Activity)的引用(不然你怎么可能通過(guò)Handler來(lái)操作Activity中的View?)。而Handler通常會(huì)伴隨著一個(gè)耗時(shí)的后臺(tái)線程(例如從網(wǎng)絡(luò)拉取圖片)一起出現(xiàn),這個(gè)后臺(tái)線程在任務(wù)執(zhí)行完畢(例如圖片下載完畢)之后,通過(guò)消息機(jī)制通知Handler,然后Handler把圖片更新到界面。然而,如果用戶在網(wǎng)絡(luò)請(qǐng)求過(guò)程中關(guān)閉了Activity,正常情況下,Activity不再被使用,它就有可能在GC檢查時(shí)被回收掉,但由于這時(shí)線程尚未執(zhí)行完,而該線程持有Handler的引用(不然它怎么發(fā)消息給Handler?),這個(gè)Handler又持有Activity的引用,就導(dǎo)致該Activity無(wú)法被回收(即內(nèi)存泄露),直到網(wǎng)絡(luò)請(qǐng)求結(jié)束(例如圖片下載完畢)。另外,如果你執(zhí)行了Handler的postDelayed()方法,該方法會(huì)將你的Handler裝入一個(gè)Message,并把這條Message推到MessageQueue中,那么在你設(shè)定的delay到達(dá)之前,會(huì)有一條MessageQueue -> Message -> Handler -> Activity的鏈,導(dǎo)致你的Activity被持有引用而無(wú)法被回收。
二、內(nèi)存泄露的危害
內(nèi)存泄露的危害就是會(huì)使虛擬機(jī)占用內(nèi)存過(guò)高,導(dǎo)致OOM(內(nèi)存溢出),程序出錯(cuò)。
對(duì)于Android應(yīng)用來(lái)說(shuō),就是你的用戶打開一個(gè)Activity,使用完之后關(guān)閉它,內(nèi)存泄露;又打開,又關(guān)閉,又泄露;幾次之后,程序占用內(nèi)存超過(guò)系統(tǒng)限制,F(xiàn)C。
三、解決方案
使用Handler導(dǎo)致內(nèi)存泄露的解決方法
方法一:通過(guò)程序邏輯來(lái)進(jìn)行保護(hù)。
1.在關(guān)閉Activity的時(shí)候停掉你的后臺(tái)線程。線程停掉了,就相當(dāng)于切斷了Handler和外部連接的線,Activity自然會(huì)在合適的時(shí)候被回收。
2.如果你的Handler是被delay的Message持有了引用,那么使用相應(yīng)的Handler的removeCallbacks()方法,把消息對(duì)象從消息隊(duì)列移除就行了。
方法二:將Handler聲明為靜態(tài)類。
PS:在Java 中,非靜態(tài)的內(nèi)部類和匿名內(nèi)部類都會(huì)隱式地持有其外部類的引用,靜態(tài)的內(nèi)部類不會(huì)持有外部類的引用。
靜態(tài)類不持有外部類的對(duì)象,所以你的Activity可以隨意被回收。由于Handler不再持有外部類對(duì)象的引用,導(dǎo)致程序不允許你在Handler中操作Activity中的對(duì)象了。所以你需要在Handler中增加一個(gè)對(duì)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) { noteBookAdapter.notifyDataSetChanged(); } } } }
PS:什么是WeakReference?
WeakReference弱引用,與強(qiáng)引用(即我們常說(shuō)的引用)相對(duì),它的特點(diǎn)是,GC在回收時(shí)會(huì)忽略掉弱引用,即就算有弱引用指向某對(duì)象,但只要該對(duì)象沒(méi)有被強(qiáng)引用指向(實(shí)際上多數(shù)時(shí)候還要求沒(méi)有軟引用,但此處軟引用的概念可以忽略),該對(duì)象就會(huì)在被GC檢查到時(shí)回收掉。對(duì)于上面的代碼,用戶在關(guān)閉Activity之后,就算后臺(tái)線程還沒(méi)結(jié)束,但由于僅有一條來(lái)自Handler的弱引用指向Activity,所以GC仍然會(huì)在檢查的時(shí)候把Activity回收掉。這樣,內(nèi)存泄露的問(wèn)題就不會(huì)出現(xiàn)了。
四、總結(jié)
android中的很多內(nèi)存泄露都是由于在Activity中使用了非靜態(tài)內(nèi)部類導(dǎo)致的,我們?cè)谑褂梅庆o態(tài)內(nèi)部類一定要格外注意,如果該靜態(tài)內(nèi)部類的實(shí)例對(duì)象的生命周期大于外部對(duì)象,那么就有可能導(dǎo)致內(nèi)存泄露,推薦使用上面介紹的靜態(tài)類和弱引用的方法解決這種問(wèn)題。
以上所述是小編給大家介紹的Android使用Handler造成內(nèi)存泄露問(wèn)題及解決方法,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
- Android 優(yōu)化Handler防止內(nèi)存泄露
- 使用Android Studio檢測(cè)內(nèi)存泄露(LeakCanary)
- Android 消息機(jī)制以及handler的內(nèi)存泄露
- 避免 Android中Context引起的內(nèi)存泄露
- Android 中Handler引起的內(nèi)存泄露
- Android垃圾回收機(jī)制解決內(nèi)存泄露問(wèn)題
- Android中Handler引起的內(nèi)存泄露問(wèn)題解決辦法
- Android編程中避免內(nèi)存泄露的方法總結(jié)
- Android App調(diào)試內(nèi)存泄露之Cursor篇
- 分析Android常見的內(nèi)存泄露和解決方案
相關(guān)文章
Android答題APP的設(shè)計(jì)與實(shí)現(xiàn)
這篇文章主要為大家詳細(xì)介紹了Android答題APP的設(shè)計(jì)與實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01android實(shí)現(xiàn)手機(jī)傳感器調(diào)用
這篇文章主要為大家詳細(xì)介紹了android實(shí)現(xiàn)手機(jī)傳感器調(diào)用,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-04-04Android 用HttpURLConnection訪問(wèn)網(wǎng)絡(luò)的方法
下面小編就為大家分享一篇Android 用HttpURLConnection訪問(wèn)網(wǎng)絡(luò)的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-01-01Android實(shí)現(xiàn)網(wǎng)易嚴(yán)選標(biāo)簽欄滑動(dòng)效果
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)網(wǎng)易嚴(yán)選標(biāo)簽欄滑動(dòng)效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-07-07Android Studio實(shí)現(xiàn)進(jìn)度條效果
這篇文章主要為大家詳細(xì)介紹了Android Studio實(shí)現(xiàn)進(jìn)度條效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-04-04Android自定義控件實(shí)現(xiàn)簡(jiǎn)單滑動(dòng)開關(guān)效果
這篇文章主要為大家詳細(xì)介紹了Android自定義控件實(shí)現(xiàn)簡(jiǎn)單滑動(dòng)開關(guān)效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-02-02詳談android 6.0 fuse文件系統(tǒng)的掛載和卸載問(wèn)題
今天小編就為大家分享一篇詳談android 6.0 fuse文件系統(tǒng)的掛載和卸載問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-08-08Android Kotlin的使用及簡(jiǎn)單實(shí)例
這篇文章主要介紹了Android Kotlin的使用及簡(jiǎn)單實(shí)例的相關(guān)資料,需要的朋友可以參考下2017-05-05