EventBus源码解析
EventBus是一个开源的事件总线项目,项目地址:EventBus
EventBus通过注册监听器和发布消息的方式来完成事件的传递,如下所示:
data:image/s3,"s3://crabby-images/00b1e/00b1e305330e02b5eac1a9d91c6f68bdebfe336f" alt=""
其中Publisher为事件的发布者,Subscriber为事件的订阅者。下图为EventBus的DEMO使用方法:
data:image/s3,"s3://crabby-images/2cf81/2cf8143f8ddb22fd10580ea5bb971aa0ebbd2739" alt=""
使用EventBus主要分为以下四步:
1、注册成为事件接受者:EventBus.getDefault().register(this)将当前类注册为事件接受者
2、注销:通过EventBus.getDefault().unregister(this)进行注销;
3、订阅事件:通过@Subscribe注解对方法进行注解,注解的方法必须有且仅有一个参数,这里我们定义了MyEvent作为订阅的事件。
4、发布事件:EventBus.getDefault().post(MyEvent)可以发布一个事件,EventBus会根据发布的消息的类型,找到该类型消息的订阅者,完成消息的传递。
@Subscribe注解
data:image/s3,"s3://crabby-images/7c84f/7c84f3f0a30ee4cabc0cf26e2735b915dd87bd6e" alt=""
@Subscribe注解的定义如上图所示,通过@Rentention将注解定义为TUNTIME,表示该注解将会保留到JVM运行时,也就是可以通过反射进行调用;通过@Target将注解定义为METHOD,表示该注解只能应用于method;在使用@Subscribe注解时,可以设置threadMode、sticky、priority属性。
threadMode:指定订阅方法运行的线程,根据其值在对应的线程调用订阅Method。
sticky:指定是否是粘性事件,如果为true,在调用register的时候,会接受到注册监听之前EventBus postSticky发送过的同一类型的消息。
priority:优先级,EventBus根据订阅者的优先级来顺序发布事件。
EventBus的主要成员
EventBus.getDefault()以单例的方式获取EventBus实例,这里重点介绍下EventBus的成员变量:
data:image/s3,"s3://crabby-images/90df4/90df4fc62cf4e6b7497baaf01279eb60eb52dba6" alt=""
1、subscriptionsByEventType
data:image/s3,"s3://crabby-images/45b9c/45b9c8c77fb099cafdfe876f4601bfccb098e94d" alt=""
是一个Map对象,Key为@Subcribe注解的方法的参数类型,对应到本文一开始的DEMO的MyEvent的Class对象,Value为Subscription列表。
data:image/s3,"s3://crabby-images/43baf/43baf8038f9b3c8a063163e4a81822cade44aaf1" alt=""
Subscription的subscriber对象指的是EventBus.getDefault().register(this)的this对象,subscriberMethod指的是@Subcribe注解的方法的Method对象。
一句话:subscriptionsByEventType保存了订阅MyEvent事件的订阅者列表。
2、typesBySubscriber
data:image/s3,"s3://crabby-images/8bb7a/8bb7a66441a5adbedc0ef2ccb11b5fc72daa5e34" alt=""
是一个Map对象,Key为EventBus.getDefault().register(this)的this对象,Value为this对象中所有的@Subcribe注解方法的参数类型,也就是订阅的事件类型。
3、stickyEvents
data:image/s3,"s3://crabby-images/44f4a/44f4aed5474d78f86a51d3488b8b4c17578fbf6d" alt=""
是一个Map对象,postSticky发送的粘性事件会保存到该Map中。
data:image/s3,"s3://crabby-images/83c8d/83c8d290310ebe19dea3008fbb425df6de104156" alt=""
Key为发送的粘性事件的Class对象,Value为粘性事件。
4、mainThreadSupport&mainThreadPoster(threadMode=MAIN 或者MAIN_ORDERED)
类型为mainThreadSupport类型为MainThreadSupport,mainThreadPoster类型为HandlerPoster。
data:image/s3,"s3://crabby-images/34b1f/34b1f0d25a8a451ac9eae6d3560026941d5aeb22" alt=""
通过Looper对象判断当前是否是主线程,创建HandlerPoster赋值给mainThreadPoster,HandlerPoster继承自Handler,handleMessage完成事件的分发,保证订阅方法运行在主线程。
5、backgroundPoster(threadMode=BACKGROUND)
和mainThreadPoster类似,实现类是BackgroundPoster,该类继承实现了Runnable接口,在异步线程调用订阅者的方法。
data:image/s3,"s3://crabby-images/30d0b/30d0bb4502e6cd4bed67137ab292b2312cd691b4" alt=""
6、asyncPoster(threadMode=ASYNC)
data:image/s3,"s3://crabby-images/00d32/00d32c973b3841572b7a4c45604213a3df510e5e" alt=""
实现类是AsyncPoster,也是通过后台线程调用订阅者方法,完成事件发布。但与BackgroundPoster有如下区别:
BackgroundPoster:如果当前线程是主线程,就调用backgroundPoster发布事件;如果当前发布时已经是异步线程,那么就直接反射运行订阅者方法,不需要开启线程了。
AsyncPoster:无论在哪个线程post事件都开启新的线程执行。
7、subscriberMethodFinder
实现类是SubscriberMethodFinder,在EventBus.getDefault().register(this)的时候查询this对象及其父类对象的所有@Subcribe注解的方法。
到这里,介绍完EventBus主要的成员后,下面分析下EventBus.getDefault().register的流程。
注册事件订阅者
EventBus.getDefault().register(this);将当前类注册为事件订阅者。
data:image/s3,"s3://crabby-images/33246/33246f953371cfa89393355513b45895feb1fe53" alt=""
register主要分为以下三步:
1、subscriber.getClass获取订阅者的Class对象
2、subscriberMethodFinder.findSubscriberMethods(subscriberClass)获取订阅者及其父类的所有@Subscribe注解的Method。
data:image/s3,"s3://crabby-images/3881a/3881a238c68740fb0b3d9460df2b0549dd1da732" alt=""
通过findUsingInfo来获取订阅者及其父类的所有订阅方法:
data:image/s3,"s3://crabby-images/e10be/e10be07f150d10610269c405ee51d17f454b2cdb" alt=""
findUsingInfo首先初始化了FindState,FindState是个什么东东?
data:image/s3,"s3://crabby-images/5eb93/5eb933728e14a36b280feaf0837b0b45862ed439" alt=""
上图描述了FindState的作用和信息,FindState用来查找和保存订阅者及其父类的订阅信息,订阅的方法保存在subscriberMethods中;subscriberInfo默认为空,所以在初始化完FindState后会循环调用findUsingReflectionInSingleClass方法,该方法通过反射来获取FindState.clazz的@Subscribe注解方法,findUsingInfo会一直循环,直到clazz已经没有符合要求的父类了。
data:image/s3,"s3://crabby-images/3dc53/3dc539da542b6bffe59e413a050c38e0b286c6f9" alt=""
在循环结束后,findState.subscriberMethods保存了订阅者及其父类的所有SubscriberMethod。
3、对每个subscriberMethod调用subscribe(subscriber, subscriberMethod)方法。
第2步获取到所有订阅方法后,就循环调用subscribe(subscriber, subscriberMethod)。
data:image/s3,"s3://crabby-images/96c16/96c16b227945efe06dbc9fa61bc71bc408a82b2b" alt=""
data:image/s3,"s3://crabby-images/9d0af/9d0af64e1b9a04be25832c0c52565566ef0216a5" alt=""
至此,register方法运行结束了,subscriptionsByEventType中保存了某个类型消息对应的Subscription列表,Subscription中包含了订阅者和订阅方法;typesBySubscriber保存了某个订阅者的所有订阅事件类型;最后也完成了sticky事件的发布。
发布事件
通过EventBus.getDefault().post(postEvent);可以发布一个事件。
data:image/s3,"s3://crabby-images/28316/2831610c1811cd33be3ac71e8ebc8e3d9cf98dca" alt=""
发布事件有以下过程:
1、获取当前的发布状态PostingThreadState
data:image/s3,"s3://crabby-images/de156/de1562b1612e12cd5c0b9628e1ea52abf92f1055" alt=""
PostingThreadState定义了当前正在发布的事件状态。
2、将事件保存到等待队列
3、循环调用postSingleEvent发布事件
data:image/s3,"s3://crabby-images/e346f/e346ff15091df4095600071db0eebe238afd8960" alt=""
postSingleEvent调用了lookupAllEventTypes()获取要发布的事件及其父类的Class对象,也就是说如果发布MyEvent事件,订阅了MyEvent事件或者其父类事件的订阅者都能收到MyEvent事件(可以这样理解,订阅了MyEvent的父类事件,MyEvent是其父类的一种类型,也应该发布)。postSingleEventForEventType完成单个事件的发布:
data:image/s3,"s3://crabby-images/ba061/ba061c4f373a55ba47b3e9be5d5d0561ff0e980b" alt=""
首先在subscriptionsByEventType获取eventClass对应的订阅列表,然后调用postToSubscription进行发布。
data:image/s3,"s3://crabby-images/7548a/7548ac1b6e627e7f50b03f46539a2665da0c2fda" alt=""
postToSubscription会根据SubscriberMethod的ThreadMode确定调用哪个Poster就进行事件发布。无论在主线程还是异步线程,最终都是调用了invokeSubscriber:
data:image/s3,"s3://crabby-images/76c62/76c62b3f371a0348b2650c983af4a9732f17cf38" alt=""
invokeSubscriber通过反射调用了subscriber的method,并传入了Event参数。
4、重置PostingThreadState状态
注销
通过EventBus.getDefault().unregister(this);可以完成注销,即订阅者不在关注任何事件了。
data:image/s3,"s3://crabby-images/752e0/752e0719ea94c441b4d19cf10f7556657031b4b5" alt=""
data:image/s3,"s3://crabby-images/85a68/85a68619c3f2ba7f653d16b9403bd914c5be0eb6" alt=""
至此,EventBus的注册、订阅消息、发布消息、注销的流程都已经分析完毕。