Handler引起的内存泄露(Activity中Handler的正确使用方法)

转载 2015年02月04日 17:15:54

如果你在Activity中定义了一个内部Handler类,如下代码:

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

然后运行Android Lint工具会有一个内存泄露警告:

This Handler class should be static or leaks might occur (com.example.ta.MainActivity.1)
Issue: Ensures that Handler classes do not hold on to a reference to an outer class
Id: HandlerLeak
In Android, Handler classes should be static or leaks might occur. Messages enqueued on the application thread’s MessageQueue also retain their target Handler. If the Handler is an inner class, its outer class will be retained as well. To avoid leaking the outer class, declare the Handler as a static nested class with a WeakReference to its outer class.

原因是:

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

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

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

  • 如果外部类是Activity,则会引起Activity泄露 。

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

要修改该问题,只需要按照Lint提示的那样,把Handler类定义为静态即可,然后通过WeakReference 来保持外部的Activity对象。

private Handler mHandler = new MyHandler(this);
private static class MyHandler extends Handler{
    private final WeakReference<Activity> mActivity;
    public MyHandler(Activity activity) {
        mActivity = new WeakReference<Activity>(activity);
    }
    @Override
    public void handleMessage(Message msg) {
        System.out.println(msg);
        if(mActivity.get() == null) {
            return;
        }
    }
}

所以,当你在Activity中使用内部类的时候,需要时刻考虑您是否可以控制该内部类的生命周期,如果不可以,则最好定义为静态内部类。

参考文章: http://blog.chengyunfeng.com/?p=468

Android Handler正确使用姿势

Android Handler正确使用姿势
  • geanwen
  • geanwen
  • 2017年01月08日 15:14
  • 737

Android中Handler的使用

Handler是Android中引入的一种让开发者参与处理线程中消息循环的机制。每个Hanlder都关联了一个线程,每个线程内部都维护了一个消息队列MessageQueue,这样Handler实际上也...

Activity内部Handler引起内存泄露的原因分析

Since this Handler is declared as an inner class, it may prevent the outer class from being garbage ...
  • sun927
  • sun927
  • 2015年01月22日 01:18
  • 1843

Handler常见用法小结

这次对Hanlder的常见用法做一个小结,巩固一下基础知识。在Handler的异步操作中,我们会使用到Handler,Message,Looper。 Handler一般用来接收Message...

Handler有何作用?如何使用?

android Handler有何作用?如何使用?

Activity与Fragment通过Handler交互

自去年七月份以后,就没有再动过笔了。为这段长达半年多的懈怠感到羞愧。唉,言归正传,上代码吧。 正常activity与fragment的交互,如果有跳转的话,就是直接用intent和bundle来进行...
  • as4399
  • as4399
  • 2017年01月31日 12:53
  • 1468

Handler最正确的写法(避免内存泄露)

/**                         Handler最正确的写法:  handler重点在  原理  和 内存泄露  *   *   * Handler的内存泄露问题:  * 目的:...

Activity已经销毁了,线程还在运行的问题

不得不说,这个问题解决了一天。。。。 原先不知道是后台线程的问题,后来发现了。是这样的,我在Activity 中加了一个线程,为了实现ProgressBar(就是进度条),我最近再做手机游戏,我想的...

Android开发常见bug及解决方法 (一)

1、音频播放时,一定要思考异常处理、如其它事件播放音频的打断处理; 2、下载资源等异步操作的监听器实现时,一定要对用到的所有成员变量判空      IDownloadResultListener ...

Android中使用Handler造成内存泄露的分析和解决

Java使用有向图机制,通过GC自动检查内存中的对象(什么时候检查由虚拟机决定),如果GC发现一个或一组对象为不可到达状态,则将该对象从内存中回收。也就是说,一个对象不被任何引用所指向,则该对象会在被...
  • jdsjlzx
  • jdsjlzx
  • 2016年05月12日 18:21
  • 5720
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Handler引起的内存泄露(Activity中Handler的正确使用方法)
举报原因:
原因补充:

(最多只允许输入30个字)