You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 

3.3 KiB

Handler 消息机制口水话

大纲

  1. 基本使用
  2. 源码分析
  3. 常见问题
  4. 实际应用
  5. 其他

Android 应用是通过消息驱动运行的,在 Android 中一切皆消息,包括触摸事件、视图的绘制、显示和刷新等。Handler 是消息机制的上层接口,平时开发中我们只会接触到 Handler 和 Message,内部还有 MessageQueue 和 Looper 两大助手共同实现消息循环系统。

Handler 的基本使用我就不多说了,不过需要注意的是内存泄露,最好使用静态内部类 + 弱引用,并且在 Activity 退出时移除消息。

Handler 消息机制比较简单:

首先通过 Handler 的 sendMessage 或 post 两种方式去发送消息,post 内部也是 sendMessage 形式,只不过把传入的 Runnable 参数包装成 Message 的 callback。然后把 Message 传给 MessageQueue 的 enqueueMessage 入队,其实就是维护一个 Message 链表,它会根据 Message 的 when 时间排序,延迟消息不过是延时时间再加上当前时间。

然后就是核心逻辑了,在 Looper 的 loop 方法,它是一个死循环,里面做了三件事,调用 MessageQueue 的 next 方法取消息,然后通过 Message 的 target 也就是 handler 去 dispatchMessage 分发消息,最后回收消息。MessageQueue 的 next 也是一个死循环,首先会调用 nativePollOnce 函数,如果没有消息或者处理时间未到,就会阻塞在 Native 层的 pollOnce 函数,睡眠时间就是从 Java 层传过来的。不过其核心实现是在 pollInner 中,如果监听的文件描述符没有发生 IO 读写事件,那么当前线程就会在 epoll_wait 中进入休眠。如果当前线程有新的消息需要处理,就会被唤醒,然后沿着之前的调用路径返回到 Java 层,最后就是对新消息进行处理。

看完源码,有两点可以参考,第一是获取 Message 最好通过 Message 的 obtain 获取,不要直接 new,因为 Message.obtain 会从缓存里面去取。第二是可以使用 IdleHandler 在消息队列空闲时提前做一些操作。这个我在项目中也有用到,在点击消息中心图标时,会跳到 h5 页面,并且在 url 后面加密拼接 uid 和 phone number,这一加密操作是同步的,所以就可以通过 IdleHandler 提前做这一操作,并且返回 false,表示只做一次。

下面在简单说下设置同步分隔栏,即 MessageQueue.postSyncBarrier()。同步分隔栏的原理其实很简单,本质上是通过创建一个 target 成员为 null 的 Message 并插入到消息队列中,这样在这个特殊的 Message 之后的消息就不会被处理了,只有当这个 Message被移除后才会继续执行之后的 Message。最经典的实现就 ViewRootImpl 调用 scheduleTraversals 方法进行视图更新时使用,它通过 Choreographer postCallback 把 Message 设置成异步消息避免被分割,把 doTraversal 封装成一个 Runnable,在真正执行时才会移除这个分隔栏。这样做的原因是,doTraversal 的操作是通过 Handler 进行处理的,然鹅这个消息队列却是主线程公用的,比如四大组件的各个生命周期的调用,显然视图绘制这个无疑是最高优先级的,所以在这之前,要确保队列中其他同步消息不会影响到它的执行。