簡(jiǎn)介
?LeakCanary是一款開(kāi)源的內(nèi)存泄露檢測(cè)工具,可以用來(lái)檢測(cè)項(xiàng)目中的Activity是否能夠被GC及時(shí)回收纽哥。
使用方式
在項(xiàng)目的build.gradle文件添加:
dependencies {
???? debugImplementation 'com.squareup.leakcanary:leakcanary-android:1.3'
???? releaseImplementation 'com.squareup.leakcanary:leakcanary-android-no-op:1.3'
}
然后在Application中調(diào)用install方法:
public class MyApplication extends Application {
????? @Override
????? public void onCreate() {
??????????? super.onCreate();
??????????? LeakCanary.install(this);
????? }
}
LeakCanary.install()方法大致流程:
LeakCanary.install() -->? 創(chuàng)建RefWatcher --> 創(chuàng)建ActivityRefWatcher --> watchActivities() --> 注冊(cè)Activity生命周期回調(diào)
當(dāng)生命周期回調(diào)onActivityDestroyed的時(shí)候,觸發(fā)ActivityRefWatcher的onActivityDestroyed調(diào)用:
ActivityRefWatcher.onActivityDestroyed() --> RefWatcher.watch() --> 在watch方法內(nèi)部,通過(guò)關(guān)注的activity的引用、隨機(jī)生成的key值微猖、一個(gè)引用隊(duì)列ReferenceQueue,? 來(lái)構(gòu)建一個(gè)KeyedWeakReference 弱引用對(duì)象。同時(shí)缘屹,將key加入retainedKeys集合凛剥。
接著采用異步的方式--- ensureGone() , 來(lái)檢測(cè)該activity是否能被GC回收
ensureGone方法內(nèi)部2個(gè)核心方法:removeWeaklyReachableReferences() 和 gone() .
知識(shí)補(bǔ)充: 使用引用隊(duì)列ReferenceQueue來(lái)創(chuàng)建WeakReference對(duì)象,當(dāng)WeakReference所引用的Object轻姿,在某一時(shí)刻變?yōu)椤叭蹩蛇_(dá)” (JVM規(guī)定的可GC回收狀態(tài))的時(shí)候犁珠,對(duì)此Object的弱引用會(huì)被清除,同時(shí)該WeakReference會(huì)入隊(duì)ReferenceQueue中互亮。
removeWeaklyReachableReferences方法犁享,就是通過(guò)遍歷取出ReferenceQueue中的KeyedWeakReference,來(lái)判定其所引用的activity是已達(dá)到GC可回收狀態(tài)豹休,同時(shí)將對(duì)應(yīng)的key從retainedKeys集合中移除炊昆。后續(xù)通過(guò)調(diào)用gone方法,返回ture威根,說(shuō)明如果該key已經(jīng)不在retainedKeys集合中凤巨,說(shuō)明key對(duì)應(yīng)的activity已達(dá)可回收狀態(tài),沒(méi)有內(nèi)存泄露風(fēng)險(xiǎn)医窿。
否則磅甩,會(huì)手動(dòng)觸發(fā)一次系統(tǒng)GC,再嘗試一次剛才的判斷流程姥卢。如果gone方法仍然返回false卷要,說(shuō)明已經(jīng)存在內(nèi)存泄露風(fēng)險(xiǎn),將會(huì)dump當(dāng)前堆內(nèi)存情況独榴,將dump file交給HeapAnalyzerService來(lái)分析:
最終通過(guò) HeapAnalyzer.checkForLeak() 來(lái) 計(jì)算出 activity實(shí)例? 到GC roots的最短強(qiáng)引用路徑僧叉。