android 内部handler引发内存泄露

一段很常见的代码: java

public class MainActivity extends Activity {
 
    private  Handler mHandler = newHandler() {
        @Override
        public void handleMessage(Message msg) {
            //TODO handle message...
        }
 
    };
 
    @TargetApi(11)
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mHandler.sendMessageDelayed(Message.obtain(), 60000);
 
        //just finish this activity
        finish();
    }
}

可是你运行lint会发现有一个内存泄露的警告。 ide

 

缘由是:
1.
当Android应用启动的时候,会先建立一个应用主线程的Looper对象,Looper实现了一个简单的消息队列,一个一个的处理里面的Message对象。主线程Looper对象在整个应用生命周期中存在。 oop

2.
当在主线程中初始化Handler时,该Handler和Looper的消息队列关联。发送到消息队列的Message会引用发送该消息的Handler对象,这样系统能够调用 Handler#handleMessage(Message) 来分发处理该消息。 this

3.
在Java中,非静态(匿名)内部类会引用外部类对象。而静态内部类不会引用外部类对象。 线程

4.
若是外部类是Activity,则会引发Activity泄露 。 code

 

具体到这个例子里边,当Activity finish后,延时消息会继续存在主线程消息队列中1分钟,而后处理消息。而该消息引用了Activity的Handler对象,而后这个Handler又引用了这个Activity。这些引用对象会保持到该消息被处理完,这样就致使该Activity对象没法被回收,从而致使了上面说的 Activity泄露。 对象

要修改该问题,只须要按照Lint提示的那样,把Handler类定义为静态便可,而后经过WeakReference 来保持外部的Activity对象。 生命周期

private Handler mHandler = new MyHandler(this);
private static class MyHandler extendsHandler{
    private final WeakReference<Activity> mActivity;
    public MyHandler(Activity activity) {
        mActivity = newWeakReference<Activity>(activity);
    }
    @Override
    public void handleMessage(Message msg) {
        System.out.println(msg);
        if(mActivity.get() == null) {
            return;
        }
    }
}
相关文章
相关标签/搜索