Android LeakCanary 检测内存泄露

内存泄漏:

指程序在申请内存后 ,无法释放已经申请的内存空间,一次内存泄漏可以忽略,但内存泄漏堆积后果很严重,无论多少内存,都会被占光

内存泄露危害:

1.内存泄露最终会导致内存溢出(OOM)

2.导致程序卡顿,应用程序莫名退出

内存泄露检测工具

  • Memory Monitor(不能精准的定位问题)
  • MAT工具 (操作复杂,学习成本高)
  • 还有许多就不一一列举了

Android中常见的内存泄露(欢迎指证与添加)

大图片也能造成OOM

hangdler造成的内存泄露解决办法

当Activity退出时消息队列中还有未处理的消息或者正在处理的消息
而消息队列中的Messager持有handler实例的引用, handler又持有activity的引用,所以导致Activity的内存资源无法及时回收,引发内存泄露

解决办法:
1.使用弱引用
2.使用静态handler内部类
3.在onDestory()方法中调用removeCallbacksAndMessagers(null);

单例造成的内存泄露解决办法

原因:单例的静态特性 使得单例的生命周期和应用的生命周期一样长,这就说明如果一个对象已经不需要使用了,而单例对象还持有该对象的引用,那么这个对象不能正常回收,就会导致内存泄露

解决办法:如果此时传入的是 Activity 的 Context,当这个 Context 所对应的 Activity 退出时,由于该 Context 的引用被单例对象所持有,其生命周期等于整个应用程序的生命周期,所以当前 Activity 退出时它的内存并不会被回收,这就造成泄漏了
所以如果此时传入的是 Application 的 Context,因为 Application 的生命周期就是整个应用的生命周期,所以这没有任何问题。

非静态内部类创建实例造成的内存泄露

将该内部类设为静态内部类或将该内部类抽取出来封装成一个单例,如果需要使用Context,请使用ApplicationContext

线程造成的内存泄露

异步任务和Runnable都是一个匿名内部类,因此它们对当前Activity都有一个隐式引用。如果Activity在销毁之前,任务还未完成,
那么将导致Activity的内存资源无法回收,造成内存泄漏

解决办法
正确的做法还是使用静态内部类的方式

资源没有关闭造成的内存泄露

如数据库 流

LeakCanary介绍

这个才是今天的正题

LeakCanary使用示例

  参考LeakCanary官方示例 https://github.com/square/leakcanary
首先我们需要在应用的 build.gradle 中,添加依赖:
<pre><code>
debugCompile 'com.squareup.leakcanary:leakcanary-android:1.5.2'
releaseCompile 'com.squareup.leakcanary:leakcanary-android-no-op:1.5.2
</code></pre>

LeakCanary依赖添加后,我们添加一个Application类

1.创建个MyApplication类
<pre><code>
public class MyApplication extends Application {
private RefWatcher refWatcher;
@Override
public void onCreate() {
super.onCreate();
refWatcher = LeakCanary.install(this);
}
}
</pre></code>

2.在AndroidManifest.xml文件中去配置这个application

 android:name=".MyApplication"

3.LeakCanary的配置就已经完成了,

然后运行应用程序除了会安装自己的App外还会安装leaks这个东西

https://github.com/square/leakcanary/raw/master/assets/icon_512.png
https://github.com/square/leakcanary/raw/master/assets/icon_512.png

4.完(万)事具备,只欠东方 下面我就举个内存泄露的例子
<pre><code>
private void initView() {
Button tv = (Button) findViewById(R.id.tv);
tv.setText("dsafsdfadsfdsafsa");
tv.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
AsyncTaskTest();
finish();
}
}
);
}
public void AsyncTaskTest() {
new AsyncTask<Void, Void, Void>() {
@Override
protected Void doInBackground(Void... params) {
SystemClock.sleep(20000000);
return null;
}
}.execute();
}
</pre></code>

5.点击按钮 稍等片刻(可能时间稍微有点长),就会出现下面这个信息了

这里写图片描述
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 内存管理的目的就是让我们在开发中怎么有效的避免我们的应用出现内存泄漏的问题。内存泄漏大家都不陌生了,简单粗俗的讲,...
    宇宙只有巴掌大阅读 2,415评论 0 12
  • 前言 对于内存泄漏,我想大家在开发中肯定都遇到过,只不过内存泄漏对我们来说并不是可见的,因为它是在堆中活动,而要想...
    EsonJack阅读 964评论 1 3
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 173,301评论 25 708
  • 我们全家都爱吃肉夹馍 可是对于外面的肉肉有点不放心 还是自己亲自动手吧 亲自和面 烙饼 亲自煮肉 做成了家人都爱吃...
    苏小柚阅读 327评论 0 1
  • 在现实的生活当中,你可能对当下的事情,很难做到百分之百的判断。但是对已经发生的事情,你的总结和完善,那才是重要的。
    泽阳9阅读 158评论 0 1