Android中內(nèi)存泄漏需要的注意點(diǎn)
內(nèi)存泄漏對每一位 Android 開發(fā)一定是司空見慣,大家或多或少都肯定有些許接觸。大家都知道,每一個(gè)手機(jī)都有一定的承載上限,多處的內(nèi)存泄漏堆積一定會堆積如山,最終出現(xiàn)內(nèi)存爆炸 OOM。
而這,也是極有可能在 Android 面試中一道常見的開放題。
內(nèi)存泄漏的根本原因是一個(gè)長生命周期的對象持有了一個(gè)短生命周期的對象。如果你對垃圾回收機(jī)制有所了解,我想這個(gè)問題基本難不住你,因?yàn)橹懒嗽?,自然不會去觸碰這些極易導(dǎo)致內(nèi)存泄漏的雷區(qū)。
該題重在積累,不需要死記硬背,自己多總結(jié)即可。
1. 長生命周期對象持有 Activity
這基本是最常見的內(nèi)存泄漏了,比如
內(nèi)部類形式使用 Handler 同時(shí)發(fā)送延時(shí)消息,或者在 Handler 里面執(zhí)行耗時(shí)任務(wù),在任務(wù)還沒完成的時(shí)候 Activity 需要銷毀。這時(shí)候由于 Handler 持有 Activity 的強(qiáng)引用導(dǎo)致 Activity 無法被回收。
同理內(nèi)部類形式的使用 AsyncTask 執(zhí)行耗時(shí)任務(wù)也會導(dǎo)致內(nèi)存泄漏的發(fā)生。
單例作為最長生命周期的對象,自然不應(yīng)該持有 Activity 從而導(dǎo)致內(nèi)存泄漏發(fā)生;
針對上面這種情況,基本不必多說了,不要使用內(nèi)部類或者匿名內(nèi)部類做這樣的處理就好了,實(shí)際上 IDE 也會彈出警告,我想大家應(yīng)該還是都知道采用靜態(tài)內(nèi)部類或者在銷毀頁面的時(shí)候使用相關(guān)方法移除處理的。
Activity 中匿名使用 Handler 實(shí)際上會導(dǎo)致 Handler 內(nèi)部類持有外部類的引用,而 SendMessage() 的時(shí)候 Message 會持有 Handler,enqueueMessage 機(jī)制又會導(dǎo)致 MeassageQueue 持有 Message。所以當(dāng)發(fā)送的是延遲消息那么 Message 并不會立即的遍歷出來處理而是阻塞到對應(yīng)的 Message 觸發(fā)時(shí)間以后再處理。那么阻塞的這段時(shí)間中頁面銷毀一定會造成內(nèi)存泄漏。
2. 各種注冊操作沒有對應(yīng)的反注冊
這一點(diǎn)基本不必多說,相信大家剛剛開始學(xué)習(xí)廣播和 Service 的時(shí)候一定對此有所接觸,然后就是比如我們常用的第三方框架 EventBus 也是一樣的。平時(shí)使用的時(shí)候注意在對應(yīng)的生命周期方法中進(jìn)行反注冊。
3. Bitmap 使用完沒有注意 recycle()
Bitmap 作為大對象,在使用完畢一定要注意調(diào)用 recycle() 進(jìn)行回收。TypedArray 、Cursor、各種流同理,一定要在最后調(diào)用自己的回收關(guān)閉方法處理。
4. WebView 使用不當(dāng)
WebView 是非常常用的控件,但稍有不注意也會導(dǎo)致內(nèi)存泄漏。內(nèi)存泄漏的場景: 很多人使用 Webview 都喜歡采用布局引用方式, 這其實(shí)也是作為內(nèi)存泄漏的一個(gè)隱患。當(dāng) Activity 被關(guān)閉時(shí),Webview 不會被 GC 馬上回收,而是提交給事務(wù),進(jìn)行隊(duì)列處理,這樣就造成了內(nèi)存泄漏, 導(dǎo)致 Webview 無法及時(shí)回收。
目前所知的比較安全的方案是:
在布局中動(dòng)態(tài)添加 WebView。
采用下面的方法。
override fun onDestroy() { webView?.apply { val parent = parent if (parent is ViewGroup) { parent.removeView(this) } stopLoading() // 退出時(shí)調(diào)用此方法,移除綁定的服務(wù),否則某些特定系統(tǒng)會報(bào)錯(cuò) settings.javaScriptEnabled = false clearHistory() removeAllViews() destroy() } }
5. 循環(huán)引用
循環(huán)引用導(dǎo)致內(nèi)存泄漏比較少見,正常來講不會有人寫出 A 持有 B,B 持有 C,C 又持有A 這樣的代碼,不過總還是需要注意。
總的來說,內(nèi)存泄漏很常見,但檢測方式也很多。我們的 Android Studio 自帶的 Monitors 就可以幫我們找到大部分內(nèi)存問題,當(dāng)然我們也可以采用譬如 LeakCanary 這樣的庫去做檢測。
相關(guān)文章
Android仿美團(tuán)下拉菜單(商品選購)實(shí)例代碼
這篇文章主要介紹了Android仿美團(tuán)下拉菜單(商品選購)實(shí)例代碼的相關(guān)資料,需要的朋友可以參考下2016-03-03Android 多線程實(shí)現(xiàn)重復(fù)啟動(dòng)與停止的服務(wù)
這篇文章主要介紹了Android 多線程實(shí)現(xiàn)重復(fù)啟動(dòng)與停止的服務(wù)的相關(guān)資料,多線程環(huán)境下為了避免死鎖,一般提倡開放調(diào)用,開放調(diào)用可以避免死鎖,它的代價(jià)是失去原子性,這里說明重復(fù)啟動(dòng)與停止的服務(wù),需要的朋友可以參考下2017-08-08淺析Android圓形進(jìn)度條ProgressBar如何實(shí)現(xiàn)固定進(jìn)度
之前遇到一個(gè)問題,發(fā)現(xiàn)Android里的圓形進(jìn)度條無法固定一個(gè)進(jìn)度,所以這篇文章就來和大家探索一下圓形進(jìn)度條ProgressBar如何實(shí)現(xiàn)固定進(jìn)度,希望對大家有所幫助2024-03-03Android編程之SurfaceView實(shí)例詳解
這篇文章主要介紹了Android編程之SurfaceView用法,簡要分析了View和SurfaceView的區(qū)別,并結(jié)合實(shí)例形式分析了SurfaceView的具體使用步驟與相關(guān)技巧,需要的朋友可以參考下2016-02-02Android 中自定義Dialog樣式的Activity點(diǎn)擊空白處隱藏軟鍵盤功能(dialog不消失)
項(xiàng)目中需要開發(fā)帶有EditText的Dialog顯示,要求在編輯完EditText時(shí),點(diǎn)擊Dilog的空白處隱藏軟鍵盤。但是Dialog不會消失。下面通過實(shí)例代碼給大家分享實(shí)現(xiàn)方法,需要的的朋友參考下吧2017-04-04android webview獲取html代碼和根據(jù)id獲取value實(shí)例
這篇文章主要介紹了android webview獲取html代碼和根據(jù)id獲取value實(shí)例,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-03-03