内存泄漏:
指程序在申请内存后 ,无法释放已经申请的内存空间,一次内存泄漏可以忽略,但内存泄漏堆积后果很严重,无论多少内存,都会被占光
内存泄露危害:
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这个东西
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>