消息總線EventBus源碼分析以及與Otto框架對比
(一).前言:
上一篇我們對EventBus的簡介和基本使用做了說明,今天我們主要深入的使用EventBus,同時會從源碼的角度對于訂閱和發送消息做分析,以及和另外的消息總線框架Otto在性能等方面做一個對比分析。
FastDev4Android框架項目地址: https://github.com/jiangqqlmj/FastDev4Android
(二).框架簡單說明:
通過上一篇文章的介紹,EventBus的使用步驟如下:
- 定義一個事件,用于EventBus的分發。
- 定義訂閱者,把該訂閱者加入到EventBus中。
- 通過EventBus.post來進行分發事件,告訴訂閱者有事情發生了。訂閱者接收到信息進行相應處理。
- 使用完成之后,訂閱者需要反注冊取消訂閱。
具體原理圖如下:
訂閱者接收到通知的時候會調用相應的函數進行處理事件,在EventBus中一般有以下四種方法來讓我們進行處理:
- onEvent
- onEventMainThread
- onEventBackground
- onEventAsync
這四個訂閱方法有很多的相似之處,但是功能上面還是有點不同的,EventBus會通過調用post方法來進行分發消息,讓訂閱者進行接收,訂閱者接收到事件消息是通過上面幾個方法來進行接收和處理的。下面我們來對這四個方法的具體使用場景做一個介紹:
- onEvent:使用該方法作為訂閱函數表示post消息事件和接收消息事件在同一個線程中。
- onEventMainThread: 該方法會在UI Main線程中運行,接收事件同時會在UI線程中運行,這樣我們就可以在該方法中直接更新UI
- onEventBackground:使用該方法,如果事件是在UI Main線程發出來,該方法會在子線程中執行,如果事件是從子線程中發出來,該onEventBackground方法會在子線程中執行。
- onEventAsync:使用該方法,會在創建新的子線程中執行onEventAsync
那么現在訂閱的函數方法有四個,我們怎么會知道具體調用哪個方法呢?OK我們看一篇文章:我們會先創建一個事件類,然后進行post發送該對象,在訂閱方法中接收,注入哪個函數的參數就是該發送過來的對象。這樣我們應該清楚了吧,那是根據傳進來的事件對象參數來進行判斷的。具體我們看實例:
(三).調用實例:
3.1.實現需求:我們現在創建三個Event事件類,第二個Activity中進行發送,在訂閱者Activity中進行接收訂閱方法如下:
/** * 收到消息 進行相關處理 * @param event */ public void onEventMainThread(TestEventFirst event) { Log.d("zttjiangqq","onEventMainThread收到消息:"+event.getMsg()); textView_one.setText(event.getMsg()); //showToastMsgShort(event.getMsg()); } /** * 收到消息 進行相關處理 * @param event */ public void onEventMainThread(TestEventSecond event) { Log.d("zttjiangqq","onEventMainThread收到消息:"+event.getMsg()); textView_two.setText(event.getMsg()); //showToastMsgShort(event.getMsg()); } /** * 收到消息 進行相關處理 * @param event */ public void onEventMainThread(TestEventThird event) { Log.d("zttjiangqq","onEventMainThread收到消息:"+event.getMsg()); textView_third.setText(event.getMsg()); //showToastMsgShort(event.getMsg()); }
3.2.演示效果如下:
(四).源碼解析:
以上主要為EventBus的主要使用,現在開始我們對于EventBus的注冊和發送兩個模塊從源碼的角度來走一下。
4.1.EventBus對象獲取:我們一般使用單例模式獲取。保證對象唯一性。
/** * 采用單例模式獲取EventBus實例對象 一般我們獲取EventBus對象 就是采用這種方式,不建議直接new * @return */ public static EventBus getDefault() { if (defaultInstance == null) { synchronized (EventBus.class) { if (defaultInstance == null) { defaultInstance = new EventBus(); } } } return defaultInstance; }
4.2.訂閱模塊:入口,進行訂閱注冊
public void register(Object subscriber) { register(subscriber, false, 0); }
- subscriber:需要注冊的訂閱者,
- sticky:是否為粘性,這邊默認為false,
- priority:事件的優先級,默認為0
下面我們來具體看一下register(subscriber, false, 0)方法具體實現的功能:
private synchronized void register(Object subscriber, boolean sticky, int priority) { List<SubscriberMethod> subscriberMethods = subscriberMethodFinder.findSubscriberMethods(subscriber.getClass()); for (SubscriberMethod subscriberMethod : subscriberMethods) { subscribe(subscriber, subscriberMethod, sticky, priority); } }
該函數中會通過findSubscriberMethods()來獲取所有訂閱的方法,具體主要的步驟我這邊已經進行注釋了
/** * 進行查找訂閱者中所有訂閱的方法 * @param subscriberClass * @return 所有訂閱的方法的集合 */ List<SubscriberMethod> findSubscriberMethods(Class<?> subscriberClass) { String key = subscriberClass.getName(); List<SubscriberMethod> subscriberMethods; //從緩存中獲取訂閱的方法,第一次使用肯定緩存中不存在 synchronized (methodCache) { subscriberMethods = methodCache.get(key); } if (subscriberMethods != null) { return subscriberMethods; } //訂閱方法的集合 subscriberMethods = new ArrayList<SubscriberMethod>(); Class<?> clazz = subscriberClass; HashMap<String, Class> eventTypesFound = new HashMap<String, Class>(); StringBuilder methodKeyBuilder = new StringBuilder(); while (clazz != null) { String name = clazz.getName(); if (name.startsWith("java.") || name.startsWith("javax.") || name.startsWith("android.")) { // Skip system classes, this just degrades performance // 這邊直接跳過了系統類,因為系統類中 普通開發者不會使用在系統類中使用EventBus,所以就忽略處理了,不然會降低性能 break; } // Starting with EventBus 2.2 we enforced methods to be public (might change with annotations again) try { // This is faster than getMethods, especially when subscribers a fat classes like Activities // 通過反射來獲取當前類中的所有方法 Method[] methods = clazz.getDeclaredMethods(); // 正式開始查詢所有訂閱的方法 filterSubscriberMethods(subscriberMethods, eventTypesFound, methodKeyBuilder, methods); } catch (Throwable th) { th.printStackTrace(); // Workaround for java.lang.NoClassDefFoundError, see https://github.com/greenrobot/EventBus/issues/149 Method[] methods = subscriberClass.getMethods(); subscriberMethods.clear(); eventTypesFound.clear(); filterSubscriberMethods(subscriberMethods, eventTypesFound, methodKeyBuilder, methods); break; } clazz = clazz.getSuperclass(); } //拋出異常,訂閱者沒有實現onEvent開頭的公共方法 if (subscriberMethods.isEmpty()) { throw new EventBusException("Subscriber " + subscriberClass + " has no public methods called " + ON_EVENT_METHOD_NAME); } else { //訂閱的方法存在,同時加入緩存 synchronized (methodCache) { methodCache.put(key, subscriberMethods); } return subscriberMethods; } }
然后調用filterSubscriberMethods()進行過濾,把訂閱方法加入到集合中
/** * 查詢訂閱的方法,查到方法,方法加入到subScriberMethods * @param subscriberMethods * @param eventTypesFound * @param methodKeyBuilder * @param methods */ private void filterSubscriberMethods(List<SubscriberMethod> subscriberMethods, HashMap<String, Class> eventTypesFound, StringBuilder methodKeyBuilder, Method[] methods) { //遍歷類中的所有方法 for (Method method : methods) { String methodName = method.getName(); //過濾onEvent開頭的方法 if (methodName.startsWith(ON_EVENT_METHOD_NAME)) { //返回方法修飾符 例如 public,private,protected int modifiers = method.getModifiers(); Class<?> methodClass = method.getDeclaringClass(); if ((modifiers & Modifier.PUBLIC) != 0 && (modifiers & MODIFIERS_IGNORE) == 0) { //訂閱方法必須為public類型 //進行獲取方法的參數類型 Class<?>[] parameterTypes = method.getParameterTypes(); if (parameterTypes.length == 1) { //進行獲取線程模式類型 ThreadMode threadMode = getThreadMode(methodClass, method, methodName); if (threadMode == null) { continue; } //取出當前傳入的訂閱者 Class<?> eventType = parameterTypes[0]; //methodKeyBuilder key ="0"."methodName".">"."eventType_Name" methodKeyBuilder.setLength(0); methodKeyBuilder.append(methodName); methodKeyBuilder.append('>').append(eventType.getName()); String methodKey = methodKeyBuilder.toString(); Class methodClassOld = eventTypesFound.put(methodKey, methodClass); if (methodClassOld == null || methodClassOld.isAssignableFrom(methodClass)) { // Only add if not already found in a sub class // 構建一個訂閱方法的對象(里面存入方法名,線程模式類型,事件類型),加入到訂閱方法集合中 subscriberMethods.add(new SubscriberMethod(method, threadMode, eventType)); } else { // Revert the put, old class is further down the class hierarchy eventTypesFound.put(methodKey, methodClassOld); } } } else if (!skipMethodVerificationForClasses.containsKey(methodClass)) { Log.d(EventBus.TAG, "Skipping method (not public, static or abstract): " + methodClass + "." + methodName); } } } }
上面已經進行獲取了所有的訂閱函數,那么現在開始就可以進行訂閱了,讓我們來查看subscribe()方法做的功能操作:
/** * 開始進行為訂閱者 注冊相關的訂閱方法 * @param subscriber 訂閱者 * @param subscriberMethod 訂閱的方法 * @param sticky 是否為粘性 * @param priority 優先級 */ private void subscribe(Object subscriber, SubscriberMethod subscriberMethod, boolean sticky, int priority) { //通過訂閱方法中進行獲取訂閱方法的類型 Class<?> eventType = subscriberMethod.eventType; //通過訂閱事件的類型 進行獲取所有的訂閱信息(有訂閱者對象,訂閱方法,優先級) CopyOnWriteArrayList<Subscription> subscriptions = subscriptionsByEventType.get(eventType); //進行創建一個訂閱者 Subscription newSubscription = new Subscription(subscriber, subscriberMethod, priority); if (subscriptions == null) { //如果當前的事件類型不存在訂閱信息,那么就創建一個訂閱信息集合 subscriptions = new CopyOnWriteArrayList<Subscription>(); //同時把當前的訂閱信息加入到該訂閱中 subscriptionsByEventType.put(eventType, subscriptions); } else { if (subscriptions.contains(newSubscription)) { //拋出異常,該訂閱者已經注冊過該事件類中 throw new EventBusException("Subscriber " + subscriber.getClass() + " already registered to event " + eventType); } } // Starting with EventBus 2.2 we enforced methods to be public (might change with annotations again) // subscriberMethod.method.setAccessible(true); // 優先級判斷,進行排序 int size = subscriptions.size(); for (int i = 0; i <= size; i++) { if (i == size || newSubscription.priority > subscriptions.get(i).priority) { subscriptions.add(i, newSubscription); break; } } //將當前的事件加入到訂閱者列表中 List<Class<?>> subscribedEvents = typesBySubscriber.get(subscriber); if (subscribedEvents == null) { subscribedEvents = new ArrayList<Class<?>>(); typesBySubscriber.put(subscriber, subscribedEvents); } subscribedEvents.add(eventType); //是否粘性判斷 if (sticky) { if (eventInheritance) { // Existing sticky events of all subclasses of eventType have to be considered. // Note: Iterating over all events may be inefficient with lots of sticky events, // thus data structure should be changed to allow a more efficient lookup // (e.g. an additional map storing sub classes of super classes: Class -> List<Class>). Set<Map.Entry<Class<?>, Object>> entries = stickyEvents.entrySet(); for (Map.Entry<Class<?>, Object> entry : entries) { Class<?> candidateEventType = entry.getKey(); if (eventType.isAssignableFrom(candidateEventType)) { Object stickyEvent = entry.getValue(); checkPostStickyEventToSubscription(newSubscription, stickyEvent); } } } else { Object stickyEvent = stickyEvents.get(eventType); checkPostStickyEventToSubscription(newSubscription, stickyEvent); } } }
OK完成以上步驟,我們就大體完成了訂閱注冊工作,下面就是需要分析一下post的流程:
主要先看post主函數:
/** * 向EventBus中發送消息事件對象 * @param event */ public void post(Object event) { PostingThreadState postingState = currentPostingThreadState.get(); //把消息加入到事件隊列中 List<Object> eventQueue = postingState.eventQueue; eventQueue.add(event); if (!postingState.isPosting) { postingState.isMainThread = Looper.getMainLooper() == Looper.myLooper(); postingState.isPosting = true; if (postingState.canceled) { throw new EventBusException("Internal error. Abort state was not reset"); } try { //當消息隊列不為空的時候,進行這正式發送消息,采用循環,把隊列中所有的消息發送出去 while (!eventQueue.isEmpty()) { postSingleEvent(eventQueue.remove(0), postingState); } } finally { postingState.isPosting = false; postingState.isMainThread = false; } } }
然后進行發送功能,調用postSingleEvent()函數方法:
//消息發送:發送單個事件消息 private void postSingleEvent(Object event, PostingThreadState postingState) throws Error { Class<?> eventClass = event.getClass(); boolean subscriptionFound = false; if (eventInheritance) { List<Class<?>> eventTypes = lookupAllEventTypes(eventClass); int countTypes = eventTypes.size(); for (int h = 0; h < countTypes; h++) { Class<?> clazz = eventTypes.get(h); subscriptionFound |= postSingleEventForEventType(event, postingState, clazz); } } else { subscriptionFound = postSingleEventForEventType(event, postingState, eventClass); } if (!subscriptionFound) { if (logNoSubscriberMessages) { Log.d(TAG, "No subscribers registered for event " + eventClass); } if (sendNoSubscriberEvent && eventClass != NoSubscriberEvent.class && eventClass != SubscriberExceptionEvent.class) { post(new NoSubscriberEvent(this, event)); } } }
接著進行消息過濾postSingleEventForEventType()方法
/** * 進行該特定的Event發送相應的消息 * @param event 事件消息 * @param postingState * @param eventClass * @return */ private boolean postSingleEventForEventType(Object event, PostingThreadState postingState, Class<?> eventClass) { CopyOnWriteArrayList<Subscription> subscriptions; synchronized (this) { subscriptions = subscriptionsByEventType.get(eventClass); } if (subscriptions != null && !subscriptions.isEmpty()) { for (Subscription subscription : subscriptions) { postingState.event = event; postingState.subscription = subscription; boolean aborted = false; try { //發生消息給訂閱者 postToSubscription(subscription, event, postingState.isMainThread); aborted = postingState.canceled; } finally { postingState.event = null; postingState.subscription = null; postingState.canceled = false; } if (aborted) { break; } } return true; } return false; }
最終這邊有一個核心的方法:postToSubscription()來進行post消息
/** * 進行發送消息,同時根據發送過來的線程類型類型,發送消息給特定的訂閱方法來進行執行 * @param subscription 訂閱者 * @param event 執行事件 * @param isMainThread 是否為主線程 */ private void postToSubscription(Subscription subscription, Object event, boolean isMainThread) { switch (subscription.subscriberMethod.threadMode) { case PostThread: //直接在本線程中調用訂閱函數 invokeSubscriber(subscription, event); break; case MainThread: if (isMainThread) { //如果是主線程,直接調用訂閱函數 invokeSubscriber(subscription, event); } else { //如果不是主線程,通過handler進行處理 mainThreadPoster.enqueue(subscription, event); } break; case BackgroundThread: if (isMainThread) { //如果是主線程,采用runnable 中調用 backgroundPoster.enqueue(subscription, event); } else { //子線程,直接調用 invokeSubscriber(subscription, event); } break; case Async: //加入到子線程中進行發送 asyncPoster.enqueue(subscription, event); break; default: throw new IllegalStateException("Unknown thread mode: " + subscription.subscriberMethod.threadMode); } }
OK,到這邊基本上完成EventBus的register和post的流程的講解,關于這個核心類EventBus的注釋過的類文件已經上傳了大家可以通過該地址進行下載: EventBus注釋過的類文件
(五).和Otto消息總線框架對比:
Otto是Android中另外一個消息總線庫,它其實是EventBus的變種,該和EventBus有一些相同的方法(例如:register,post,unregister…),但是這兩者之間也有一些不同之處如下:
EventBus |
Otto |
|
聲明事件處理方法 | 命名約定 | 注解 |
事件繼承 |
YES |
YES |
訂閱繼承 |
YES |
NO |
緩存事件 | YES,sticky events | NO |
事件生產 |
NO |
YES |
子線程事件傳輸 | YES(Default) |
YES |
主線程事件傳輸 |
YES |
NO |
后臺線程事件傳輸 |
YES |
NO |
異步線程事件傳輸 |
YES |
NO |
除了以上功能不同以外,這邊還有性能上面的差異。為了測試性能問題,我們clone當前EventBus項目的時候,會發現有一個EventBusPerformance項目,我們可以使用的不同場景來比較。
基于下表結果顯示,每一個測試方面 EventBus的性能都大于Otto
EventBus |
Otto |
|
在Android2.3模擬器發送1000個事件 | 快70% | |
S3 Android4.0系統,發送1000個事件 | 快110% | |
Android2.3模擬器,注冊1000個訂閱者 | 快10% | |
S3 Android4.0系統,注冊1000個訂閱者 | 快70% | |
Android2.3模擬器,注冊訂閱者冷啟動 | 快350% | |
S3 Android4.04 注冊訂閱者冷啟動 | 幾乎一樣 |
通過對比發現EventBus無論在功能上面還是性能上面,遠遠超過Otto消息總線框架,所以我們建議使用EventBus消息總線框架。
到此我們的EventBus專題內容已經全部講完了,相信大家在這個專題中能對EventBus會有一個比較全面的了解,同時也能夠簡單的了解實現的原理以及相關邏輯。
我們的項目已經配置集成了消息總線EventBus的例子.歡迎大家去Github站點進行clone或者下載瀏覽: https://github.com/jiangqqlmj/FastDev4Android 同時歡迎大家star和fork整個開源快速開發框架項目~
尊重原創,轉載請注明:From Sky丶清(http://www.lcode.org) 侵權必究!
關注我的訂閱號(codedev123),每天分享移動開發技術(Android/IOS),項目管理以及博客文章!(歡迎關注,第一時間推送精彩文章)
關注我的微博,可以獲得更多精彩內容