Android中的LeakCanary的原理詳解
場景:最新的leakCanary2.8.1:
debugImplementation 'com.squareup.leakcanary:leakcanary-android:2.8.1'
原理:首先就是我們在引入最新的依賴包,什麼都不用幹瞭,因為他的初始化在清單文件中註冊瞭contentProvider(),把初始化放到瞭這裡面的onCreate()去初始化瞭,在初始化的過程中,他會用application監聽觀察對象activity、fragment等對象的生命周期的變化,當執行銷毀的生命周期,他就會用對應ActivityWatch—>ObjectWatch來觀察你這個銷毀的對象,那怎麼觀察呢?將對象加入到弱引用對象,並把這個弱引用和一個引用隊列Queue來綁定(同時把這個弱引用先添加到一個map的觀察列表),這樣的話當主動Gc的時候,如果沒有泄露,就會回收這個activity觀察對象,並會把這個弱引用加入到引用隊列中去,我們就可以去判斷這個引用隊列有沒有值,有就代表沒泄露,否則為queue.poll()取出來為null就泄露瞭,最後會把這個泄露對象的弱引用添加到一個set集合,叫做retained objects,最終會使用shark庫(原來是haha分析庫)去查詢泄露的地方生成Dump文件,把分析結果發通知給開發者。
通知點擊:告知retained objects—點擊–>Dumping Heap—自動–>Analyzing heap
如何看這個分析的結果:
上面兩個圖就是這個泄露對象的引用鏈關系,最後就是存在泄露的對象LoginActivity,那為什麼泄露就得往上去尋找,發現是在Dialog單例中持有瞭context(即LoginActivity對象),及時走瞭destory也不會銷毀這個對象,因為被GcRoot一直持有。
這是引起內存泄漏的代碼:
object LoadingDialog { //內部生成的時候,根據INSTANCE 看起來感覺是靜態,因為可以LoadingDialog.show() //其實是偽靜態 fun show() { } //這種寫法才是靜態方法 @JvmStatic fun show2() { } private var dialog:Dialog?=null fun show(context: Context) { cancel() dialog = Dialog(context) dialog?.setContentView(R.layout.dialog_loading) dialog?.setCancelable(false) dialog?.setCanceledOnTouchOutside(false) dialog?.show() } fun cancel() { dialog?.dismiss() } }
解決就是,把dialog用完要置為null
fun cancel() { dialog?.dismiss() dialog = null; }
這樣leakCanary就不會通知泄露點瞭。
到此這篇關於Android中的LeakCanary的原理詳解的文章就介紹到這瞭,更多相關Android LeakCanary內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- Android 進階實現性能優化之OOM與Leakcanary詳解原理
- Android LeakCanary的使用方法介紹
- Android實現加載圈
- Android內存泄漏的原因及解決技巧
- Android自定義仿ios加載彈窗