怎么解决由于Handler引起的Memory Leak
既然知道了,Memory Leak的根本原因是Handler持有Context,导致Context无法释放。那么解决的办法就是断开Handler和Context之间的连接,让Handler不再持有Context的引用。
大概的方法有:
将隐性匿名类Handler变成static的内部类,由于static的内部类,使用的使用不需要外部类的实例,所以static的内部类和外部类是没有联系的,从而不持有外部类的引用;
将隐性匿名类写成一个单独的类(top-level-class),这样Handler和Context之间就没有联系了。
将Handler做成static 内部类
public class HandlerTestActivity extends Activity {
private final TestHandler mHandler = new TestHandler(this);
static class TestHandler extends Handler {
WeakReference<HandlerTestActivity> mActivity;
MyHandler(HandlerTestActivity activity) {
mActivity = new WeakReference<HandlerTestActivity>(activity);
}
@Override
public void handleMessage(Message msg) {
HandlerTestActivity activity = mActivity.get();
//一定要注意null判断
if(activity != null){
//do something
}
}
};
}
上面代码中,将TestHandler定义成static的内部类,这样断开了Handler和Context之间的联系,从而Handler不再持有Context的实例。
Handler通过弱引用保持Context的实例。弱引用的特点是,当系统GC的时候,是可以自动释放掉的,这样在系统GC的时候,Context也就被释放掉了,不会引起Memory Leak。
Runnable怎么处理
public class HandlerTestActivity extends Activity {
private Runnable mRun = new Runnable() {
@Override
public void run() {
// ... do something
}
}
}
由于上面的Runnable也存在Handler的Memory Leak的问题。原因是Runnable持有Context的实例。
所以Runnable的解决办法也是讲Runnable和Context断开连接。将Runnable声明为static的即可。
public class HandlerTestActivity extends Activity {
private static Runnable mRun = new Runnable() {
@Override
public void run() {
// ... do something
}
}
}