EventBus-源碼解析

基于eventbus 2.4.0
項(xiàng)目中很多地方用到EventBus
大體的使用 就是這樣的幾段代碼

// 注冊(cè)
EventBus.getDefault().register(this);
// 發(fā)送
 EventBus.getDefault().post(new Event());
// 響應(yīng)
public void onEventMainThread(Event event) {
}
// 取消注冊(cè)
EventBus.getDefault().unregister(this);

EventBus.getDefault() 利用單例模式獲取的是EventBus實(shí)例。
register 方法

// 采用的是訂閱者模式
private synchronized void register(Object subscriber, boolean sticky, int priority) {
        // findSubscriberMethods 通過反射獲取subscriber的 onEventMainThread方法
        List<SubscriberMethod> subscriberMethods = this.subscriberMethodFinder.findSubscriberMethods(subscriber.getClass());
        // 獲取迭代器
        Iterator i$ = subscriberMethods.iterator();
        // 遍歷訂閱
        while(i$.hasNext()) {
            SubscriberMethod subscriberMethod = (SubscriberMethod)i$.next();
            // 訂閱
            this.subscribe(subscriber, subscriberMethod, sticky, priority);
        }

    }

SubscriberMethod 類中保存了 響應(yīng)Method

訂閱方法里是怎么做的授账?大體思路是按照優(yōu)先級(jí)把一個(gè)個(gè)Subscription對(duì)象(包含注冊(cè)對(duì)象)放入EventType對(duì)應(yīng)的List中迄沫,再把注冊(cè)對(duì)象的EventType放到對(duì)應(yīng)的List中开睡。

private void subscribe(Object subscriber, SubscriberMethod subscriberMethod, boolean sticky, int priority) {
        Class<?> eventType = subscriberMethod.eventType;
        CopyOnWriteArrayList<Subscription> subscriptions = (CopyOnWriteArrayList)this.subscriptionsByEventType.get(eventType);
         // 利用訂閱者對(duì)象和響應(yīng)方法創(chuàng)建一個(gè)新的對(duì)象
        Subscription newSubscription = new Subscription(subscriber, subscriberMethod, priority);
        if (subscriptions == null) {
            subscriptions = new CopyOnWriteArrayList();
            this.subscriptionsByEventType.put(eventType, subscriptions);
        } else if (subscriptions.contains(newSubscription)) {
            // 一個(gè)對(duì)象只允許注冊(cè)一次
            throw new EventBusException("Subscriber " + subscriber.getClass() + " already registered to event " + eventType);
        }

        int size = subscriptions.size();
        // 添加到此事件對(duì)應(yīng)的list里
        for(int i = 0; i <= size; ++i) {
            if (i == size || newSubscription.priority > ((Subscription)subscriptions.get(i)).priority) {
                subscriptions.add(i, newSubscription);
                break;
            }
        }
         // 通過注冊(cè)的對(duì)象 獲取注冊(cè)的event類對(duì)象
        List<Class<?>> subscribedEvents = (List)this.typesBySubscriber.get(subscriber);
        if (subscribedEvents == null) {
            subscribedEvents = new ArrayList();
            this.typesBySubscriber.put(subscriber, subscribedEvents);
        }

        ((List)subscribedEvents).add(eventType);
        if (sticky) {
            // sticky 默認(rèn)false 暫時(shí)不研究 
            Map var11 = this.stickyEvents;
            Object stickyEvent;
            synchronized(this.stickyEvents) {
                stickyEvent = this.stickyEvents.get(eventType);
            }

            if (stickyEvent != null) {
                this.postToSubscription(newSubscription, stickyEvent, Looper.getMainLooper() == Looper.myLooper());
            }
        }

    }

取消注冊(cè)大體就是反過來清空纷责。
復(fù)雜的是發(fā)送事件的過程吆寨,涉及到線程切換推姻。

public void post(Object event) {
        EventBus.PostingThreadState postingState = (EventBus.PostingThreadState)this.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()) {
                     // // 不斷循環(huán)來發(fā)送事件
                    this.postSingleEvent(eventQueue.remove(0), postingState);
                }
            } finally {
                postingState.isPosting = false;
                postingState.isMainThread = false;
            }
        }

    }

重點(diǎn)看 postSingleEvent 主要是把注冊(cè)對(duì)象的EventType對(duì)應(yīng)的List中取出窃款,然后根據(jù)不同的線程進(jìn)行分發(fā)课兄。

private void postSingleEvent(Object event, EventBus.PostingThreadState postingState) throws Error {
        Class<?> eventClass = event.getClass();
        boolean subscriptionFound = false;
        if (this.eventInheritance) {
            // 遍歷父類event 默認(rèn)false
            List<Class<?>> eventTypes = this.lookupAllEventTypes(eventClass);
            int countTypes = eventTypes.size();

            for(int h = 0; h < countTypes; ++h) {
                Class<?> clazz = (Class)eventTypes.get(h);
                subscriptionFound |= this.postSingleEventForEventType(event, postingState, clazz);
            }
        } else {
            // 繼續(xù)調(diào)用postSingleEventForEventType執(zhí)行
            subscriptionFound = this.postSingleEventForEventType(event, postingState, eventClass);
        }

        if (!subscriptionFound) {
            if (this.logNoSubscriberMessages) {
                Log.d(TAG, "No subscribers registered for event " + eventClass);
            }

            if (this.sendNoSubscriberEvent && eventClass != NoSubscriberEvent.class && eventClass != SubscriberExceptionEvent.class) {
                this.post(new NoSubscriberEvent(this, event));
            }
        }

    }

重點(diǎn)是 postSingleEventForEventType 根據(jù)不同的線程分發(fā)

private boolean postSingleEventForEventType(Object event, EventBus.PostingThreadState postingState, Class<?> eventClass) {
        CopyOnWriteArrayList subscriptions;
        synchronized(this) {
             // 獲取event對(duì)應(yīng)的 所有注冊(cè)過的對(duì)象
            subscriptions = (CopyOnWriteArrayList)this.subscriptionsByEventType.get(eventClass);
        }

        if (subscriptions != null && !subscriptions.isEmpty()) {
            Iterator i$ = subscriptions.iterator();

            while(i$.hasNext()) {
                 // 遍歷分發(fā)事件執(zhí)行postToSubscription
                Subscription subscription = (Subscription)i$.next();
                postingState.event = event;
                postingState.subscription = subscription;
                boolean aborted = false;

                try {
                    this.postToSubscription(subscription, event, postingState.isMainThread);
                    aborted = postingState.canceled;
                } finally {
                    postingState.event = null;
                    postingState.subscription = null;
                    postingState.canceled = false;
                }

                if (aborted) {
                    break;
                }
            }

            return true;
        } else {
            return false;
        }
    }

重點(diǎn)是 postToSubscription 根據(jù)訂閱方法指定的threadMode信息來執(zhí)行不同的分發(fā)。

 private void postToSubscription(Subscription subscription, Object event, boolean isMainThread) {
        switch(subscription.subscriberMethod.threadMode) {
        case PostThread:
            this.invokeSubscriber(subscription, event);
            break;
        case MainThread:
            if (isMainThread) {
                this.invokeSubscriber(subscription, event);
            } else {
                this.mainThreadPoster.enqueue(subscription, event);
            }
            break;
        case BackgroundThread:
            if (isMainThread) {
                this.backgroundPoster.enqueue(subscription, event);
            } else {
                this.invokeSubscriber(subscription, event);
            }
            break;
        case Async:
            this.asyncPoster.enqueue(subscription, event);
            break;
        default:
            throw new IllegalStateException("Unknown thread mode: " + subscription.subscriberMethod.threadMode);
        }

    }

mainThreadPoster繼承自Handler晨继,當(dāng)調(diào)用它的enqueue方法的時(shí)候烟阐,發(fā)送一個(gè)事件并在它自身的handleMessage方法中從隊(duì)列中取值并進(jìn)行處理,從而達(dá)到在主線程中分發(fā)事件的目的。backgroundPoster也是同理蜒茄。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末唉擂,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子檀葛,更是在濱河造成了極大的恐慌玩祟,老刑警劉巖,帶你破解...
    沈念sama閱讀 207,113評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件屿聋,死亡現(xiàn)場(chǎng)離奇詭異空扎,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)润讥,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評(píng)論 2 381
  • 文/潘曉璐 我一進(jìn)店門转锈,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人楚殿,你說我怎么就攤上這事撮慨。” “怎么了脆粥?”我有些...
    開封第一講書人閱讀 153,340評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵砌溺,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我冠绢,道長(zhǎng),這世上最難降的妖魔是什么常潮? 我笑而不...
    開封第一講書人閱讀 55,449評(píng)論 1 279
  • 正文 為了忘掉前任弟胀,我火速辦了婚禮,結(jié)果婚禮上喊式,老公的妹妹穿的比我還像新娘孵户。我一直安慰自己,他們只是感情好岔留,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,445評(píng)論 5 374
  • 文/花漫 我一把揭開白布夏哭。 她就那樣靜靜地躺著,像睡著了一般献联。 火紅的嫁衣襯著肌膚如雪竖配。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,166評(píng)論 1 284
  • 那天里逆,我揣著相機(jī)與錄音进胯,去河邊找鬼。 笑死原押,一個(gè)胖子當(dāng)著我的面吹牛胁镐,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 38,442評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼盯漂,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼颇玷!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起就缆,我...
    開封第一講書人閱讀 37,105評(píng)論 0 261
  • 序言:老撾萬榮一對(duì)情侶失蹤帖渠,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后违崇,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體阿弃,經(jīng)...
    沈念sama閱讀 43,601評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,066評(píng)論 2 325
  • 正文 我和宋清朗相戀三年羞延,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了渣淳。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,161評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡伴箩,死狀恐怖入愧,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情嗤谚,我是刑警寧澤棺蛛,帶...
    沈念sama閱讀 33,792評(píng)論 4 323
  • 正文 年R本政府宣布,位于F島的核電站巩步,受9級(jí)特大地震影響旁赊,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜椅野,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,351評(píng)論 3 307
  • 文/蒙蒙 一终畅、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧竟闪,春花似錦离福、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至理朋,卻和暖如春絮识,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背嗽上。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評(píng)論 1 261
  • 我被黑心中介騙來泰國(guó)打工笋除, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人炸裆。 一個(gè)月前我還...
    沈念sama閱讀 45,618評(píng)論 2 355
  • 正文 我出身青樓垃它,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子国拇,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,916評(píng)論 2 344