EventBus的设计理念是基于观察者模式的,能够参考设计模式(1)—观察者模式先来了解该设计模式。设计模式
EventBus的使用是很是简单的,首先你要添加Guava
的依赖到本身的项目中。这里咱们经过一个最基本的例子来讲明EveentBus
是如何使用的。数组
public static void main(String...args) {
// 定义一个EventBus对象,这里的Joker是该对象的id
EventBus eventBus = new EventBus("Joker");
// 向上述EventBus对象中注册一个监听对象
eventBus.register(new EventListener());
// 使用EventBus发布一个事件,该事件会给通知到全部注册的监听者
eventBus.post(new Event("Hello every listener, joke begins..."));
}
// 事件,监听者监听的事件的包装对象
public static class Event {
public String message;
Event(String message) {
this.message = message;
}
}
// 监听者
public static class EventListener {
// 监听的方法,必须使用注解声明,且只能有一个参数,实际触发一个事件的时候会根据参数类型触发方法
@Subscribe
public void listen(Event event) {
System.out.println("Event listener 1 event.message = " + event.message);
}
}
复制代码
首先,这里咱们封装了一个事件对象Event
,一个监听者对象EventListener
。而后,咱们用EventBus
的构造方法建立了一个EventBus
实例,并将上述监听者实例注册进去。而后,咱们使用上述EventBus
实例发布一个事件Event
。而后,以上注册的监听者中的使用@Subscribe
注解声明而且只有一个Event
类型的参数的方法将会在触发事件的时候被触发。缓存
总结:从上面的使用中,咱们能够看出,EventBus与观察者模式不一样的地方在于:当注册了一个监听者的时候,只有当某个方法使用了@Subscribe
注解声明而且参数与发布的事件类型匹配,那么这个方法才会被触发。这就是说,同一个监听者能够监听多种类型的事件,也能够在屡次监听同一个事件。安全
好了,经过上面的例子,咱们了解了EventBus最基本的使用方法。下面咱们来分析一下在Guava
中是如何为咱们实现这个API的。不过,首先,咱们仍是先试着考虑一下本身设计这个API的时候如何设计,而且提出几个问题,而后带着问题到源码中寻找答案。数据结构
假如要咱们去设计这样一个API,最简单的方式就是在观察者模式上进行拓展:每次调用EventBus.post()
方法的时候,会对全部的观察者对象进行遍历,而后获取它们所有的方法,判断该方法是否使用了@Subscribe
而且方法的参数类型是否与post()
方法发布的事件类型一致,若是一致的话,那么咱们就使用反射来触发这个方法。在观察者模式中,每一个观察者都要实现一个接口,发布事件的时候,咱们只要调用接口的方法就行,可是EventBus把这个限制设定得更加宽泛,也就是监听者无需实现任何接口,只要方法使用了注解而且参数匹配便可。异步
从上面的分析中能够看出,这里面不只要对全部的监听者进行遍历,还要对它们的方法进行遍历,找到了匹配的方法以后又要使用反射来触发这个方法。首先,当注册的监听者数量比较多的时候,链式调用的效率就不高;而后咱们又要使用反射来触发匹配的方法,这样效率确定又低了一些。那么在Guava
的EventBus
中是如何解决这两个问题的?ide
另外还要注意下下文中的观察者
和监听者
的不一样,监听者用来指咱们使用EventBus.register()
注册的对象,观察者是EventBus中的对象Subscriber
,后者封装了一个监听者的全部的信息,好比监听的方法等等。 通常咱们是不会直接操做Subscriber
对象的,它的访问权限也只在EventBus的包中可访问。工具
首先,当咱们使用new
初始化一个EventBus的时候,实际都会调用到下面的这个方法:源码分析
EventBus(String identifier, Executor executor, Dispatcher dispatcher, SubscriberExceptionHandler exceptionHandler) {
this.subscribers = new SubscriberRegistry(this);
this.identifier = (String)Preconditions.checkNotNull(identifier);
this.executor = (Executor)Preconditions.checkNotNull(executor);
this.dispatcher = (Dispatcher)Preconditions.checkNotNull(dispatcher);
this.exceptionHandler = (SubscriberExceptionHandler)Preconditions.checkNotNull(exceptionHandler);
}
复制代码
这里的identifier
是一个字符串类型,相似于EventBus的id; subscribers
是SubscriberRegistry类型的,实际上EventBus在添加、移除和遍历观察者的时候都会使用该实例的方法,全部的观察者信息也都维护在该实例中; executor
是事件分发过程当中使用到的线程池,能够本身实现; dispatcher
是Dispatcher类型的子类,用来在发布事件的时候分发消息给监听者,它有几个默认的实现,分别针对不一样的分发方式; exceptionHandler
是SubscriberExceptionHandler类型的,它用来处理异常信息,在默认的EventBus实现中,会在出现异常的时候打印出log,固然咱们也能够定义本身的异常处理策咯。post
因此,从上面的分析中能够看出,若是咱们想要了解EventBus是如何注册和取消注册以及如何遍从来触发事件的,就应该从SubscriberRegistry
入手。确实,我的也认为,这个类的实现也是EventBus中最精彩的部分。
根据2.1中的分析,咱们须要在EventBus中维护几个映射,以便在发布事件的时候找到并通知全部的监听者,首先是事件类型->观察者列表
的映射。 上面咱们也说过,EventBus中发布事件是针对各个方法的,咱们将一个事件对应的类型信息和方法信息等都维护在一个对象中,在EventBus中就是观察者Subscriber
。 而后,经过事件类型映射到观察者列表,当发布事件的时候,只要根据事件类型到列表中寻找全部的观察者并触发监听方法便可。 在SubscriberRegistry中经过以下数据结构来完成这一映射:
private final ConcurrentMap<Class<?>, CopyOnWriteArraySet<Subscriber>> subscribers = Maps.newConcurrentMap();
复制代码
从上面的定义形式中咱们能够看出,这里使用的是事件的Class类型映射到Subscriber列表的。这里的Subscriber列表使用的是Java中的CopyOnWriteArraySet集合, 它底层使用了CopyOnWriteArrayList,并对其进行了封装,也就是在基本的集合上面增长了去重的操做。这是一种适用于读多写少场景的集合,在读取数据的时候不会加锁, 写入数据的时候进行加锁,而且会进行一次数组拷贝。
既然,咱们已经知道了在SubscriberRegistry内部会在注册的时候向以上数据结构中插入映射,那么咱们能够具体看下它是如何完成这一操做的。
在分析register()
方法以前,咱们先看下SubscriberRegistry内部常用的几个方法,它们的原理与咱们上面提出的问题息息相关。 首先是findAllSubscribers()
方法,它用来获取指定监听者对应的所有观察者集合。下面是它的代码:
private Multimap<Class<?>, Subscriber> findAllSubscribers(Object listener) {
// 建立一个哈希表
Multimap<Class<?>, Subscriber> methodsInListener = HashMultimap.create();
// 获取监听者的类型
Class<?> clazz = listener.getClass();
// 获取上述监听者的所有监听方法
UnmodifiableIterator var4 = getAnnotatedMethods(clazz).iterator(); // 1
// 遍历上述方法,而且根据方法和类型参数建立观察者并将其插入到映射表中
while(var4.hasNext()) {
Method method = (Method)var4.next();
Class<?>[] parameterTypes = method.getParameterTypes();
// 事件类型
Class<?> eventType = parameterTypes[0];
methodsInListener.put(eventType, Subscriber.create(this.bus, listener, method));
}
return methodsInListener;
}
复制代码
这里注意一下Multimap
数据结构,它是Guava中提供的集合结构,与普通的哈希表不一样的地方在于,它能够完成一对多操做。这里用来存储事件类型到观察者的一对多映射。 注意下1处的代码,咱们上面也提到过,当新注册监听者的时候,用反射获取所有方法并进行判断的过程很是浪费性能,而这里就是这个问题的答案:
这里getAnnotatedMethods()
方法会尝试从subscriberMethodsCache
中获取全部的注册监听的方法(即便用了注解而且只有一个参数),下面是这个方法的定义:
private static ImmutableList<Method> getAnnotatedMethods(Class<?> clazz) {
return (ImmutableList)subscriberMethodsCache.getUnchecked(clazz);
}
复制代码
这里的subscriberMethodsCache
的定义是:
private static final LoadingCache<Class<?>, ImmutableList<Method>> subscriberMethodsCache = CacheBuilder.newBuilder().weakKeys().build(new CacheLoader<Class<?>, ImmutableList<Method>>() {
public ImmutableList<Method> load(Class<?> concreteClass) throws Exception { // 2
return SubscriberRegistry.getAnnotatedMethodsNotCached(concreteClass);
}
});
复制代码
这里的做用机制是:当使用subscriberMethodsCache.getUnchecked(clazz)
获取指定监听者中的方法的时候会先尝试从缓存中进行获取,若是缓存中不存在就会执行2处的代码, 调用SubscriberRegistry中的getAnnotatedMethodsNotCached()
方法获取这些监听方法。这里咱们省去该方法的定义,具体能够看下源码中的定于,其实就是使用反射并完成一些校验,并不复杂。
这样,咱们就分析完了findAllSubscribers()
方法,整理一下:当注册监听者的时候,首先会拿到该监听者的类型,而后从缓存中尝试获取该监听者对应的全部监听方法,若是没有的话就遍历该类的方法进行获取,并添加到缓存中; 而后,会遍历上述拿到的方法集合,根据事件的类型(从方法参数得知)和监听者等信息建立一个观察者,并将事件类型-观察者
键值对插入到一个一对多映射表中并返回。
下面,咱们看下EventBus中的register()
方法的代码:
void register(Object listener) {
// 获取事件类型-观察者映射表
Multimap<Class<?>, Subscriber> listenerMethods = this.findAllSubscribers(listener);
Collection eventMethodsInListener;
CopyOnWriteArraySet eventSubscribers;
// 遍历上述映射表并将新注册的观察者映射表添加到全局的subscribers中
for(Iterator var3 = listenerMethods.asMap().entrySet().iterator(); var3.hasNext(); eventSubscribers.addAll(eventMethodsInListener)) {
Entry<Class<?>, Collection<Subscriber>> entry = (Entry)var3.next();
Class<?> eventType = (Class)entry.getKey();
eventMethodsInListener = (Collection)entry.getValue();
eventSubscribers = (CopyOnWriteArraySet)this.subscribers.get(eventType);
// 若是指定事件对应的观察者列表不存在就建立一个新的
if (eventSubscribers == null) {
CopyOnWriteArraySet<Subscriber> newSet = new CopyOnWriteArraySet();
eventSubscribers = (CopyOnWriteArraySet)MoreObjects.firstNonNull(this.subscribers.putIfAbsent(eventType, newSet), newSet);
}
}
}
复制代码
SubscriberRegistry中的register()
方法与unregister()
方法相似,咱们不进行说明。下面看下当调用EventBus.post()
方法的时候的逻辑。下面是其代码:
public void post(Object event) {
// 调用SubscriberRegistry的getSubscribers方法获取该事件对应的所有观察者
Iterator<Subscriber> eventSubscribers = this.subscribers.getSubscribers(event);
if (eventSubscribers.hasNext()) {
// 使用Dispatcher对事件进行分发
this.dispatcher.dispatch(event, eventSubscribers);
} else if (!(event instanceof DeadEvent)) {
this.post(new DeadEvent(this, event));
}
}
复制代码
从上面的代码能够看出,实际上当调用EventBus.post()
方法的时候回先用SubscriberRegistry的getSubscribers方法获取该事件对应的所有观察者,因此咱们须要先看下这个逻辑。 如下是该方法的定义:
Iterator<Subscriber> getSubscribers(Object event) {
// 获取事件类型的全部父类型和自身构成的集合
ImmutableSet<Class<?>> eventTypes = flattenHierarchy(event.getClass()); // 3
List<Iterator<Subscriber>> subscriberIterators = Lists.newArrayListWithCapacity(eventTypes.size());
UnmodifiableIterator var4 = eventTypes.iterator();
// 遍历上述事件类型,并从subscribers中获取全部的观察者列表
while(var4.hasNext()) {
Class<?> eventType = (Class)var4.next();
CopyOnWriteArraySet<Subscriber> eventSubscribers = (CopyOnWriteArraySet)this.subscribers.get(eventType);
if (eventSubscribers != null) {
subscriberIterators.add(eventSubscribers.iterator());
}
}
return Iterators.concat(subscriberIterators.iterator());
}
复制代码
这里注意如下3处的代码,它用来获取当前事件的全部的父类包含自身的类型构成的集合,也就是说,加入咱们触发了一个Interger类型的事件,那么Number和Object等类型的监听方法都能接收到这个事件并触发。这里的逻辑很简单,就是根据事件的类型,找到它及其全部的父类的类型对应的观察者并返回。
接下来咱们看真正的分发事件的逻辑是什么样的。
从EventBus.post()
方法能够看出,当咱们使用Dispatcher进行事件分发的时候,须要将当前的事件和全部的观察者做为参数传入到方法中。而后,在方法的内部进行分发操做。最终某个监听者的监听方法是使用反射进行触发的,这部分逻辑在Subscriber
内部,而Dispatcher是事件分发的方式的策略接口。EventBus中提供了3个默认的Dispatcher实现,分别用于不一样场景的事件分发:
ImmediateDispatcher
:直接在当前线程中遍历全部的观察者并进行事件分发;LegacyAsyncDispatcher
:异步方法,存在两个循环,一先一后,前者用于不断往全局的队列中塞入封装的观察者对象,后者用于不断从队列中取出观察者对象进行事件分发;实际上,EventBus有个字类AsyncEventBus就是用该分发器进行事件分发的。PerThreadQueuedDispatcher
:这种分发器使用了两个线程局部变量进行控制,当dispatch()
方法被调用的时候,会先获取当前线程的观察者队列,并将传入的观察者列表传入到该队列中;而后经过一个布尔类型的线程局部变量,判断当前线程是否正在进行分发操做,若是没有在进行分发操做,就经过遍历上述队列进行事件分发。上述三个分发器内部最终都会调用Subscriber的dispatchEvent()
方法进行事件分发:
final void dispatchEvent(final Object event) {
// 使用指定的执行器执行任务
this.executor.execute(new Runnable() {
public void run() {
try {
// 使用反射触发监听方法
Subscriber.this.invokeSubscriberMethod(event);
} catch (InvocationTargetException var2) {
// 使用EventBus内部的SubscriberExceptionHandler处理异常
Subscriber.this.bus.handleSubscriberException(var2.getCause(), Subscriber.this.context(event));
}
}
});
}
复制代码
上述方法中的executor
是执行器,它是经过EventBus
获取到的;处理异常的SubscriberExceptionHandler类型也是经过EventBus
获取到的。(原来EventBus中的构造方法中的字段是在这里用到的!)至于反射触发方法调用并无太复杂的逻辑。
另外还要注意下Subscriber还有一个字类SynchronizedSubscriber,它与通常的Subscriber的不一样就在于它的反射触发调用的方法被sychronized
关键字修饰,也就是它的触发方法是加锁的、线程安全的。
至此,咱们已经完成了EventBus的源码分析。简单总结一下:
EventBus中维护了三个缓存和四个映射:
观察者Subscriber内部封装了监听者和监听方法,能够直接反射触发。而若是是映射到监听者的话,还要判断监听者的方法的类型来进行触发。我的以为这个设计是很是棒的,由于咱们无需再在EventBus中维护一个映射的缓存了,由于Subscriber中已经完成了这个一对一的映射。
每次使用EventBus注册和取消注册监听者的时候,都会先从缓存中进行获取,不是每一次都会用到反射的,这能够提高获取的效率,也解答了咱们一开始提出的效率的问题。当使用反射触发方法的调用貌似是不可避免的了。
最后,EventBus中使用了很是多的数据结构,好比MultiMap、CopyOnWriteArraySet等,还有一些缓存和映射的工具库,这些大部分都来自于Guava。
看了EventBus的实现,由衷地感受Google的工程师真牛!而Guava中还有许多更加丰富的内容值得咱们去挖掘!
了解线程局部遍历能够参考下个人另外一篇博文:ThreadLocal的使用及其源码实现