Android中內存泄露的注意點

文章轉自公衆號nanchen的每日一問


內存泄漏對每一位 Android開發一定是司空見慣,大家或多或少都肯定有些許接觸。大家都知道,每一個手機都有一定的承載上限,多處的內存泄漏堆積一定會堆積如山,最終出現內存爆炸 OOM。

而這,也是極有可能在 Android 面試中一道常見的開放題。

內存泄漏的根本原因是一個長生命週期的對象持有了一個短生命週期的對象。如果你對垃圾回收機制有所瞭解,我想這個問題基本難不住你,因爲知道了原理,自然不會去觸碰這些極易導致內存泄漏的雷區。

該題重在積累,不需要死記硬背,自己多總結即可。

1.長生命週期對象持有 Activity

這基本是最常見的內存泄漏了,比如

  • 內部類形式使用 Handler 同時發送延時消息,或者在 Handler 裏面執行耗時任務,在任務還沒完成的時候 Activity 需要銷燬。這時候由於 Handler 持有 Activity 的強引用導致 Activity 無法被回收。
  • 同理內部類形式的使用 AsyncTask 執行耗時任務也會導致內存泄漏的發生。
  • 單例作爲最長生命週期的對象,自然不應該持有 Activity 從而導致內存泄漏發生;

針對上面這種情況,基本不必多說了,不要使用內部類或者匿名內部類做這樣的處理就好了,實際上 IDE 也會彈出警告,我想大家應該還是都知道採用靜態內部類或者在銷燬頁面的時候使用相關方法移除處理的。

Activity 中匿名使用 Handler實際上會導致 Handler內部類持有外部類的引用,而 SendMessage()的時候Message 會持有 HandlerenqueueMessage機制又會導致 MeassageQueue 持有Message。所以當發送的是延遲消息那麼Message 並不會立即的遍歷出來處理而是阻塞到對應的 Message 觸發時間以後再處理。那麼阻塞的這段時間中頁面銷燬一定會造成內存泄漏。

2. 各種註冊操作沒有對應的反註冊

這一點基本不必多說,相信大家剛剛開始學習廣播和 Service 的時候一定對此有所接觸,然後就是比如我們常用的第三方框架 EventBus 也是一樣的。平時使用的時候注意在對應的生命週期方法中進行反註冊。

3. Bitmap 使用完沒有注意 recycle()

Bitmap 作爲大對象,在使用完畢一定要注意調用 recycle() 進行回收。TypedArray、Cursor、各種流同理,一定要在最後調用自己的回收關閉方法處理。

4. WebView 使用不當

WebView 是非常常用的控件,但稍有不注意也會導致內存泄漏。內存泄漏的場景: 很多人使用 Webview 都喜歡採用佈局引用方式, 這其實也是作爲內存泄漏的一個隱患。當 Activity 被關閉時,Webview 不會被 GC 馬上回收,而是提交給事務,進行隊列處理,這樣就造成了內存泄漏, 導致 Webview 無法及時回收。

目前所知的比較安全的方案是:

  • 在佈局中動態添加 WebView。
  • 採用下面的方法。
override fun onDestroy() {
    webView?.apply {
        val parent = parent
        if (parent is ViewGroup) {
            parent.removeView(this)
        }
        stopLoading()
        // 退出時調用此方法,移除綁定的服務,否則某些特定系統會報錯
        settings.javaScriptEnabled = false
        clearHistory()
        removeAllViews()
        destroy()
    }
}

5. 循環引用

循環引用導致內存泄漏比較少見,正常來講不會有人寫出 A 持有 B,B 持有 C,C 又持有A 這樣的代碼,不過總還是需要注意。



總的來說,內存泄漏很常見,但檢測方式也很多。我們的 Android Studio 自帶的 Monitors 就可以幫我們找到大部分內存問題,當然我們也可以採用譬如 LeakCanary 這樣的庫去做檢測。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章