LeakCanary内存泄漏分析工具

前言:Square公司基于MAT开源了LeakCanary,具体git地址:https://github.com/square/leakcanary,18781 star,相当好用

使用步骤

Step1:     在相应工程中的build.gradle 中加入如下代码

 dependencies {

                              debugCompile 'com.squareup.leakcanary:leakcanary-android:1.3'

                               releaseCompile 'com.squareup.leakcanary:leakcanary-android-no-op:1.3'

                         }

注:releaseCompile 'com.squareup.leakcanary:leakcanary-android-no-op:1.3'可能编译不过,需要clean一下工程,重新再编译

Step2:   在application中加入代码如下(注:整个工程配置的Application,AndroidMenifest)


Myapplication中需要加入如下代码


LeakCanary.isInAnalyzerProcess(this)判断是否能找到对应的package

LeakCanary.install(this) this传入的必须是application,具体看源码如下


Step3:准备工作做好后,将应用安装到手机后,会对应生成一个Leaks的应用,这时在自己应用上进行各种操作,横竖屏,销毁,再打开等操作

leaks应用会提示“Dumping memory app will freeze.Brrrr.”,这时候leaks应用会在后台遍历所以activity 内存泄漏,看源码,应该把所有activity进入
List,然后遍历分析。
注意这个只能对activity进行监控

如果是fragement,需要在fragment的onDestroy()方法中使用    RefWatcher refWatcher = MyApplication.getRefWatcher(getActivity());  

Step4:接下来就是等待Leaks应用的notification了,接下来果然不负众望,手机出现通知了,内存终于泄漏了。。。

迫不及待的点击进去看,发现内存泄漏的不少,两个代码,四个泄漏,如下图

然后点击进入看详情,具体是哪里泄漏,如下图


分析:途中reference表示引用了某个实例,SplashActivity这个引用被AsyncTask占用,也就是强引用,AsyncTask引起问题原因如下

开启线程后,未结束,此时用户又一次,甚至多次开启线程,导致多次请求。 

解决方式:将线程写为静态static。

当用户开启线程后,退出界面,多次进入。由于线程持有Activity的变量的实例,导致Activity无法被回收,从而导致内存泄漏 

解决方式:采用弱引用的方式,将线程与Activity进行解耦。

强引用是什么:this:代表一条狗,AsyncTask:是有个人甲需要这只狗来提醒有没有坏人,并且锁住狗。强引用:锁住了狗狗,

但这只狗并不属于甲

最后狗主人回来了,呼唤狗狗回家,但是链子锁住了狗,狗狗回不了家。甲也不需要狗狗,狗主人也呼唤不会狗狗,狗狗就在堆栈中释放

不了。

Step5:看下上图中具体如何拴住的狗狗,看看下图代码


非静态内部类,强引用this,导致this得不到释放,这时候将 代码改成非静态内部类


如果doInbackGround中继续对this引用,还可能内存泄漏。更好的办法可以使用弱引用避免


结尾:无论是AsyncTask还是Handler、Thread、单例等,经常使用非静态内部类的地方,请使用若引用,每次都去找内存泄漏的地方,还不如避免内存泄漏

你可能感兴趣的:(LeakCanary内存泄漏分析工具)