Android中Memory Leak原因分析及解決辦法
在Android開發(fā)過(guò)程中,我們經(jīng)常碰到的情況就是在我們不清楚為什么情況下,程序突然出現(xiàn)Crash了。其中有一類日志相信大家都經(jīng)常碰到過(guò),這類日志就是OOM相關(guān)的日志。這類日志除了我們知道的Bitmap操作的時(shí)候會(huì)經(jīng)常導(dǎo)致,還有一種隱藏的較深的原因就是內(nèi)存泄露(Memory Leak)。
內(nèi)存泄露產(chǎn)生原因和影響: 原因:當(dāng)一個(gè)Object不再需要的時(shí)候,本該被GC回收時(shí),但是因?yàn)榱硪粋€(gè)正在使用的Object持有它導(dǎo)致不能正常的被回收,本該被回收的對(duì)象不能回收,還存留在堆內(nèi)存中,此時(shí)就產(chǎn)生了內(nèi)存泄露。影響:Android系統(tǒng)能夠?yàn)槊總€(gè)應(yīng)用程序分配的內(nèi)存是有限的,當(dāng)一個(gè)應(yīng)用程序中產(chǎn)生的內(nèi)存泄露過(guò)多的時(shí)候,會(huì)難免導(dǎo)致應(yīng)用程序需要的內(nèi)存超過(guò)限額而導(dǎo)致內(nèi)存溢出使得應(yīng)用程序崩潰。 內(nèi)存泄露的檢測(cè)方式
推薦使用 LeakCanary 工具來(lái)檢測(cè)應(yīng)用程序是否存在內(nèi)存泄露。LeakCanary是由 Square 開源的一款輕量級(jí)的第三方內(nèi)存泄漏檢測(cè)工具,當(dāng)檢測(cè)到程序中產(chǎn)生內(nèi)存泄漏時(shí),它將以最直觀的方式告訴我們哪里產(chǎn)生了內(nèi)存泄漏和導(dǎo)致誰(shuí)泄漏了而不能被回收。
如何避免內(nèi)存泄露
1、合理使用單例模式。
單例的靜態(tài)特性使得其生命周期和應(yīng)用的生命周期一樣長(zhǎng)。
如圖,我們先聲明一個(gè)單例對(duì)象:
然后在Activity使用的時(shí)候,習(xí)慣性的傳一個(gè)this:
集成了LeakCanary后測(cè)試,發(fā)現(xiàn)內(nèi)存泄露了:
解決方案:
一般情況下,改成如下圖的寫法就可以了,因?yàn)閱卫纳芷诤蛻?yīng)用的一樣長(zhǎng),這樣就防止了內(nèi)存泄漏。:
總結(jié):?jiǎn)卫J皆斐尚孤┑脑蚴菗碛懈L(zhǎng)生命周期的對(duì)象持有短生命周期對(duì)象的強(qiáng)引用。
2、使用資源時(shí)注意資源的關(guān)閉
一般情況下,容易產(chǎn)生內(nèi)存泄露的資源主要為:File,Cursor,Stream,Bitmap,BroadcastReceiver等,這些資源在使用時(shí)建議及時(shí)關(guān)閉,否則當(dāng)這些資源沒有及時(shí)回收的時(shí)候,內(nèi)存泄露也就產(chǎn)生了。針對(duì)這些資源使用,給如下建議:
BroadcastReceiver 在register之后,需要在適當(dāng)?shù)臅r(shí)機(jī)unregister Cursor、Stream、File 這類資源類型的對(duì)象往往會(huì)使用一些Cache,所以我們?cè)诓皇褂玫臅r(shí)候,應(yīng)該及時(shí)關(guān)閉,以便Cache被及時(shí)回收。如果我們僅僅把它的引用設(shè)置為null,而不去關(guān)閉他們,往往會(huì)造成內(nèi)存泄露。一般建議是先close()后置為null。 Bitmap在不使用的時(shí)候,調(diào)用recycle()方法。目前 Android 2.3版本以后,不需要我們手動(dòng)去這樣做的,這里也就是簡(jiǎn)單的交代一下。
3、 合理使用Handler避免內(nèi)存泄露
在我們使用Handler的時(shí)候,經(jīng)??吹骄庉嬈魈崾疚覀僅andler可能會(huì)造成內(nèi)存泄露,一般在這種情況下,我們可以將Handler獨(dú)立出來(lái)或者使用靜態(tài)內(nèi)部類,這樣可以避免內(nèi)存泄露。
這樣做的原因是:非靜態(tài)內(nèi)部類會(huì)潛在的持有它所屬的外部類的引用,但是靜態(tài)內(nèi)部類是不會(huì)的。
4、 合理的使用WeakReference來(lái)引用外部類的成員變量
我們可以使用WeakReference來(lái)規(guī)避好多潛在的內(nèi)存泄露的問(wèn)題,但是并不表明WeakReference就是解決內(nèi)存泄露的金鑰匙。是否使用WeakReference主要取決于對(duì)當(dāng)前對(duì)問(wèn)題的理解,這需要我們對(duì)問(wèn)題的的建模思想。
- Android 內(nèi)存溢出和內(nèi)存泄漏的問(wèn)題
- Android避免內(nèi)存溢出(Out of Memory)方法匯總
- Android 使用幀動(dòng)畫內(nèi)存溢出解決方案
- Android編程內(nèi)存溢出與防范方法淺析
- android 解決ViewPager加載大量圖片內(nèi)存溢出問(wèn)題
- Android編程之內(nèi)存溢出解決方案(OOM)實(shí)例總結(jié)
- Android加載圖片內(nèi)存溢出問(wèn)題解決方法
- android內(nèi)存及內(nèi)存溢出分析詳解
- Android 異步獲取網(wǎng)絡(luò)圖片并處理導(dǎo)致內(nèi)存溢出問(wèn)題解決方法
相關(guān)文章
Android實(shí)現(xiàn)微信加號(hào)菜單模式
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)微信加號(hào)菜單模式,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-08-08Android仿微信對(duì)話列表滑動(dòng)刪除效果
這篇文章主要為大家詳細(xì)介紹了Android仿微信對(duì)話列表滑動(dòng)刪除效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-08-08詳解Android的網(wǎng)絡(luò)數(shù)據(jù)存儲(chǔ)
LeanCloud是一種簡(jiǎn)單高效的數(shù)據(jù)和文件存儲(chǔ)服務(wù),本文主要介紹了利用LeanCloud來(lái)進(jìn)行網(wǎng)絡(luò)數(shù)據(jù)的存儲(chǔ)的實(shí)現(xiàn)方法。具有很好的參考價(jià)值,需要的朋友一起來(lái)看下吧2016-12-12android SQLite數(shù)據(jù)庫(kù)總結(jié)
本文主要介紹了android SQLite數(shù)據(jù)庫(kù)的相關(guān)知識(shí)。具有一定的參考價(jià)值,下面跟著小編一起來(lái)看下吧2017-01-01Android編程基于自定義控件實(shí)現(xiàn)時(shí)鐘功能的方法
這篇文章主要介紹了Android編程基于自定義控件實(shí)現(xiàn)時(shí)鐘功能的方法,結(jié)合實(shí)例形式詳細(xì)分析了Android自定義控件的定義及時(shí)鐘功能相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2018-03-03Android自定義日歷Calender代碼實(shí)現(xiàn)
這篇文章主要為大家詳細(xì)介紹了Android自定義日歷Calender實(shí)現(xiàn)代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-09-09