HandlerThread

2021-07-05  本文已影响0人  纳兰沫

HandlerThread本质上是通过继承Thread类和封装Handler类的使用,从而使得创建新线程和其他线程进行通信变得更加方便易用

1.工作原理

内部原理 = Thread类 + Handler类机制
通过继承Thread类 快速的创建一个带有Looper对象的新工作线程
通过封装Handler类 快速的创建Handler与其他线程进行通信

        HandlerThread handlerThread = new HandlerThread("handlerThread");
        //启动线程
        handlerThread.start();
        //创建工作线程和复写handleMessage
        //关联HandlerThread的Looper对象  实现消息处理操作与其他线程进行通信
        //消息处理操作的(handleMessage) 的执行线程 = handlerThread所创建的工作线程中执行
        Handler workHandler = new Handler( handlerThread.getLooper()) {
            @Override
            public void handleMessage(@NonNull Message msg) {
                //消息处理
                super.handleMessage(msg);
            }
        };
        //使用工作线程Handler向工作线程的消息队列发送消息
        Message msg = Message.obtain();
        msg.what = 2;
        msg.obj = "B";
        //通过Handler发送消息到其绑定的消息队列
        workHandler.sendMessage(msg);
        //结束线程 即停止线程的消息循环
        handlerThread.quit();

当连续执行事件时,发现是按照顺序一个个执行,是因为使用HandlerThread时只是开了一个工作线程,多次执行事件,只是将消息发送到消息队列,等到派发消息给Handler再进行对应的操作

2.Handler内存泄露

2.1原因

1.存在"未被处理/正处理的消息" -> Handler实例 -> 外部类的引用关系
2.当Handler消息队列还有未处理的消息或正在处理消息时,而外部类需销毁,就会使得外部类无法被垃圾回收器(GC)回收,从而造成内存泄露

2.2 解决方案

a.静态内部类 + 弱引用

 private  static class FHandler extends Handler {
        private WeakReference<Activity> reference;

        //在构造方法中传入需要持有的Activity实例
        public FHandler(Activity activity) {
            reference = new WeakReference<Activity>(activity);
        }

        @Override
        public void handleMessage(@NonNull Message msg) {
            switch (msg.what) {
                case 1:
                    break;
                default:
                    break;
            }
        }
  }

b.当外部类结束生命周期时,清空Handler内消息队列

   @Override
    protected void onDestroy() {
        super.onDestroy();
        mHandler.removeCallbacksAndMessages(null);
        // 外部类Activity生命周期结束时,同时清空消息队列 & 结束Handler生命周期
    }

为了保证Handler中消息队列中的所有消息都能被执行,此处推荐使用解决方案1解决内存泄露问题,即 静态内部类 + 弱引用的方式

上一篇 下一篇

猜你喜欢

热点阅读