Android技术知识

探究EventBus粘性事件实现机制

2022-05-30  本文已影响0人  搬砖小老弟

作者:长安皈故里
转载地址:https://juejin.cn/post/7102815596621856799

众所周知,EventBus是支持粘性事件的,即可以先发送粘性事件,然后再注册,代码如下:

  1. 粘性事件观察者
@Subscribe(threadMode = ThreadMode.MAIN, sticky = true)
fun registerEventBus(o: Any) {    
}
  1. 发送粘性事件
EventBus.getDefault().postSticky(Any())
  1. 注册EventBus
EventBus.getDefault().register(this)

接下来我们就来探究下EventBus的粘性事件是如何实现的。

postSticky()内部机制

  1. 如果是发送的粘性事件,会添加到stickyEvents中,看下这个属性的实现:

可以看到这个属性是一个Map集合,其中key为事件类型的class对象,value为对应的事件类型。

  1. 继续看下post(Event)方法:
  1. 首先将这个粘性事件添加到PostingThreadState(线程私有)的eventQueue集合中

  2. 通过isMainThread方法判断当前是否为主线程,最终会调用到我们熟悉的Looper.getMainLooper() == Looper.myLooper()进行判断

  3. 循环遍历eventQueue队列,不断的取出集合元素进行分发,看下postSinleEvent()方法如何实现:

  1. 如果eventInheritance为true,会查找当前发送的粘性事件类型的父类型,并返回查找到的集合

  2. 接下来就会调用postSingleEventForEventType()方法来进行最终粘性事件的分发,即通知通过@Subscribe注解注册的粘性事件观察者,看下具体实现:

  1. 调用subscriptionsByEventType获取注册该事件类型的所有订阅方法,但是由于这个时候我们是先发送的粘性事件再注册EventBus,而subscriptionsByEventType中集合元素的填充实在注册EventBus发生的,所以通过subscriptionsByEventType获取到的subscriptions将是null的,所以接下来肯定不会走下面的if代码块中的逻辑了。

postSticky()小结

上面这么多代码逻辑,其实只干了一件事,就是将这个粘性事件添加到了stickyEvents这个集合中。之后的逻辑虽多,但和粘性事件没啥关系。

register内部机制

  1. findSubscriberMethods()这个方法里面的逻辑就不带大家进行分析了,总之就干了一件事情:

查找当前类通过@Subscribe注册的所有事件订阅方法,并返回一个List<SubscriberMethod>集合,其中SubscriberMethod就是对每个注册的订阅方法和当前注册类的封装

  1. subscribe这个方法是关键,深入探究下:

从上面可以看到,最终是通过反射来实现的订阅了粘性事件方法的执行。

register小结

该方法最终会判断当前是否存在注册EventBus前发送的粘性事件,且当前注册类中存在订阅该事件类型的方法,然后立即执行。

总结

以上就是EventBus粘性事件的内部实现机制,总体来说不算复杂,大家看着文章跟着源码一步步分析应该就很容易理解这部分实现逻辑了。

上一篇下一篇

猜你喜欢

热点阅读