如何避免使用Handler造成的内存泄漏

一、什么是内存泄漏?

对象已经没有被应用程序使用,由于引用或者其它垃圾回收器不能或者无法回收它们。

二、Activity中使用Handler为什么会造成泄漏

1.非静态的内部类和匿名内部类都会隐式地持有其外部类的引用
2.new Handler(){}是匿名内部内构造对象的格式
new 父类构造器(参数列表)|实现接口()  
    {  
     //匿名内部类的类体部分  
    }

一般的匿部类构造Handler

  /**
     * 匿名内部类构造Handeler对象,默认持有外部类引用
     */
    private Handler mHandler=new Handler()
    {
        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
        }
    };

三、如何避免使用Handler造成内存泄漏

1.声明静态内部内和弱引用

 private MyHandler myHandler=new MyHandler(this);
    /**
     *1. 声明静态内部类(静态的内部类不会持有外部类的引用)
     * 2.声明弱引用(GC在回收时会忽略掉弱引用,直接回收)
     */
    static class MyHandler extends Handler {
        WeakReference<Activity> mActivityReference;
        MyHandler(Activity activity) {
            //弱引用
            mActivityReference= new WeakReference<Activity>(activity);
        }
        @Override
        public void handleMessage(Message msg) {
            switch (msg.what)
            {
                case 1:
                    final Activity activity = mActivityReference.get();
                    /**
                     * 执行些界面操作,更新等
                     *
                     */
                    Log.w("leakHandler","MyHandler没有释放,还在执行任务哦~o~:");
                    break;
            }

        }
    }


2.onDestroy()的时候调用Handler的removeCallbacksAndMessages(null),清除所有回调和消息

    @Override
    protected void onDestroy() {
        super.onDestroy();
        /**
         * 移除回调和消息
         */
        if (myHandler != null)
            myHandler.removeCallbacksAndMessages(null);
        /**
         * leakCanary去观察myHandler是否泄露了
         */
        //MyApp.getRefWatcher(this).watch(myHandler);

    }


  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
问题:Handler造成内存泄漏Android开发中,Handler是用于处理消息队列和线程间通信的一种机制。然而,如果不正确地使用Handler,可能会导致内存泄漏的问题。下面是一些可能导致Handler内存泄漏的情况: 1. 静态内部类持有外部类引用:当将Handler声明为静态内部类时,它会持有外部类的引用。这意味着即使Activity已经销毁,Handler仍然可以访问该Activity的实例,从而导致无法回收Activity的内存。 解决方法:将Handler声明为非静态内部类或匿名内部类。这样,Handler不会默认持有外部类的引用,而是通过传递参数的方式来获取对Activity的引用。 示例代码: ```java public class MyActivity extends Activity { private Handler mHandler; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); mHandler = new Handler(new Handler.Callback() { @Override public boolean handleMessage(Message msg) { // 处理消息 return false; } }); } @Override protected void onDestroy() { super.onDestroy(); // 在Activity销毁时清除消息队列中的消息,以防止内存泄漏 mHandler.removeCallbacksAndMessages(null); } } ``` 2. 在Activity中创建并持有Handler对象:如果在Activity中直接创建并持有一个Handler对象,而没有将其与特定的线程关联,那么这个Handler对象将成为GC Root,导致Activity无法被回收,从而引发内存泄漏。 解决方法:将Handler与特定的线程关联。可以通过调用Thread类的start()方法或者使用AsyncTask等异步任务类来实现。 示例代码: ```java public class MyActivity extends Activity { private Thread mThread; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); mThread = new Thread(new MyRunnable()); mThread.start(); // 将Handler与线程关联,使其成为线程的运行对象 } private class MyRunnable implements Runnable { @Override public void run() { Handler mHandler = new Handler(Looper.getMainLooper()) { @Override public void handleMessage(Message msg) { // 在指定线程中处理消息,避免内存泄漏问题 } }; // 在此处执行其他操作... } } @Override protected void onDestroy() { super.onDestroy(); // 在Activity销毁时停止线程,确保资源及时释放,避免内存泄漏问题 mThread.interrupt(); // 通过中断线程来停止其执行,释放资源 } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值