一.什么是内存泄漏
Java内部使用有向图机制,通过GC自动检查内存中的对象,检测时间由虚拟机决定.如果GC发现一个或一组对象为不可达状态,就会对其进行清理,从内存中回收.换句话说,一个对象如果不被任何一个引用所指向,那么就会在GC监测到的时候对其进行回收;另外还有一种情况,就是一组对象之间互相引用,但这些对象都没有来自外部的引用,也属于不可达状态,会被回收.
private Handler handler = new Handler(){
public void handleMessage(Message msg){
if(msg.what == 100){
adapter.notifyDataSetChanged();
}
}
};
当使用内部类来创建一个Handler的时候,该对象会隐式的含有一个外部类对象的引用(通常是Activity,不然怎么可以操作其View).而Handler通常会伴随一个耗时的后台进程,比如访问网络加载一张图片,之后通过Handler发送消息,更新UI.但是如果在请求网络的时候用户关闭了该Activity,正常情况下,该Activity不再被使用,属于不可达状态,会被GC发现并回收.但由于后台进程没有执行完(带有Handler的引用),这个Handler又带有Activity的引用,所有该Activity不属于不可达状态,不会被GC回收(内存泄漏),直到网络请求完成.或者你调用了Handler的postDelayed()方法,在delay到达之前,MessageQueue中含有Message对象,而这个对象中又含有该Handler的引用,导致Activity也无法被回收.
二.内存泄漏与内存溢出
内存泄漏会导致虚拟机内存占用率过高,导致OOM(内存溢出).
三.解决办法
通常情况下有两种解决办法(推荐使用第二种,一劳永逸)
- 通过逻辑进行保护,避免泄漏
- 声明静态类Handler
方法一:
在Activity关闭的时候停止后台进程,相应的Handler调用removeCallbacks()方法.
方法二:
由于Handler是静态的,不再持有外部类对象的引用,导致程序不允许你在Handler中操作Activity中的对象了.所以你需要在Handler中增加一个对Activity的弱引用(WeakReference),GC回收内存时会忽略若引用,视为不可达状态.
static class MyHandler extends Handler{
WeakReference<Activity> mWeakReference;
public MyHandler(Activity activity) {
mWeakReference = new WeakReference<Activity>(activity);
}
@Override
public void handleMessage(Message msg){
final Activity activity = mWeakReference.get();
if(activity != null){
if (msg.what == 100)
adapter.notifyDataSetChanged();
}
}
}
四.总结
Android中大多数的内存泄漏都是由于在Activity中使用了非静态内部类导致的,我们在使用非静态内部类一定要格外注意,如果该内部类实例对象的生命周期大于外部对象,那么就有可能导致内存泄漏.