eventbus java_AndroidEventBus/EventBus.java at master · hehonghui/AndroidEventBus · GitHub

/*

* Copyright (C) 2015 Mr.Simple

*

* Licensed under the Apache License, Version 2.0 (the "License");

* you may not use this file except in compliance with the License.

* You may obtain a copy of the License at

*

* http://www.apache.org/licenses/LICENSE-2.0

*

* Unless required by applicable law or agreed to in writing, software

* distributed under the License is distributed on an "AS IS" BASIS,

* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.

* See the License for the specific language governing permissions and

* limitations under the License.

*/

package org.simple.eventbus;

import android.util.Log;

import org.simple.eventbus.handler.AsyncEventHandler;

import org.simple.eventbus.handler.DefaultEventHandler;

import org.simple.eventbus.handler.EventHandler;

import org.simple.eventbus.handler.UIThreadEventHandler;

import org.simple.eventbus.matchpolicy.DefaultMatchPolicy;

import org.simple.eventbus.matchpolicy.MatchPolicy;

import java.util.ArrayList;

import java.util.Collections;

import java.util.Iterator;

import java.util.LinkedList;

import java.util.List;

import java.util.Map;

import java.util.Queue;

import java.util.concurrent.ConcurrentHashMap;

import java.util.concurrent.ConcurrentLinkedQueue;

import java.util.concurrent.CopyOnWriteArrayList;

/**

*

* EventBus是AndroidEventBus框架的核心类,也是用户的入口类.它存储了用户注册的订阅者信息和方法,

* 事件类型和该事件对应的tag标识一个种类的事件{@see EventType},每一种事件对应有一个或者多个订阅者{@see Subscription}

* ,订阅者中的订阅函数通过{@see Subcriber}注解来标识tag和线程模型,这样使得用户体检较为友好,代码也更加整洁.

*

* 用户需要在发布事件前通过@{@see #register(Object)}方法将订阅者注册到EventBus中,EventBus会解析该订阅者中使用了

* {@see Subcriber}标识的函数,并且将它们以{@see EventType}为key,以{@see Subscription}

* 列表为value存储在map中. 当用户post一个事件时通过事件到map中找到对应的订阅者,然后按照订阅函数的线程模型将函数执行在对应的线程中.

*

* 最后在不在需要订阅事件时,应该调用{@see #unregister(Object)}函数注销该对象,避免内存泄露!

* 例如在Activity或者Fragment的onDestory函数中注销对Activity或者Fragment的订阅.

*

* 注意 : 如果发布的事件的参数类型是订阅的事件参数的子类,订阅函数默认也会被执行。例如你在订阅函数中订阅的是List类型的事件,

* 但是在发布时发布的是ArrayList的事件,

* 因此List是一个泛型抽象,而ArrayList才是具体的实现

* ,因此这种情况下订阅函数也会被执行。如果你需要订阅函数能够接收到的事件类型必须严格匹配 ,你可以构造一个EventBusConfig对象,

* 然后设置MatchPolicy然后在使用事件总线之前使用该EventBusConfig来初始化事件总线.

* EventBusConfig config = new EventBusConfig();

config.setMatchPolicy(new StrictMatchPolicy());

EventBus.getDefault().initWithConfig(config);

*

*

* @author mrsimple

*/

public final class EventBus {

/**

* default descriptor

*/

private static final String DESCRIPTOR = EventBus.class.getSimpleName();

/**

* 事件总线描述符描述符

*/

private String mDesc = DESCRIPTOR;

/**

* EventType-Subcriptions map

*/

private final Map> mSubcriberMap = new ConcurrentHashMap>();

/**

*

*/

private List mStickyEvents = Collections

.synchronizedList(new LinkedList());

/**

* the thread local event queue, every single thread has it's own queue.

*/

ThreadLocal> mLocalEvents = new ThreadLocal>() {

protected java.util.Queue initialValue() {

return new ConcurrentLinkedQueue();

};

};

/**

* the event dispatcher

*/

EventDispatcher mDispatcher = new EventDispatcher();

/**

* the subscriber method hunter, find all of the subscriber's methods

* annotated with @Subcriber

*/

SubsciberMethodHunter mMethodHunter = new SubsciberMethodHunter(mSubcriberMap);

/**

* The Default EventBus instance

*/

private static EventBus sDefaultBus;

/**

* private Constructor

*/

private EventBus() {

this(DESCRIPTOR);

}

/**

* constructor with desc

*

* @param desc the descriptor of eventbus

*/

public EventBus(String desc) {

mDesc = desc;

}

/**

* @return

*/

public static EventBus getDefault() {

if (sDefaultBus == null) {

synchronized (EventBus.class) {

if (sDefaultBus == null) {

sDefaultBus = new EventBus();

}

}

}

return sDefaultBus;

}

/**

* register a subscriber into the mSubcriberMap, the key is subscriber's

* method's name and tag which annotated with {@see Subcriber}, the value is

* a list of Subscription.

*

* @param subscriber the target subscriber

*/

public void register(Object subscriber) {

if (subscriber == null) {

return;

}

synchronized (this) {

mMethodHunter.findSubcribeMethods(subscriber);

}

}

/**

* 以sticky的形式注册,则会在注册成功之后迭代所有的sticky事件

*

* @param subscriber

*/

public void registerSticky(Object subscriber) {

this.register(subscriber);

// 处理sticky事件

mDispatcher.dispatchStickyEvents(subscriber);

}

/**

* @param subscriber

*/

public void unregister(Object subscriber) {

if (subscriber == null) {

return;

}

synchronized (this) {

mMethodHunter.removeMethodsFromMap(subscriber);

}

}

/**

* post a event

*

* @param event

*/

public void post(Object event) {

post(event, EventType.DEFAULT_TAG);

}

/**

* 发布事件

*

* @param event 要发布的事件

* @param tag 事件的tag, 类似于BroadcastReceiver的action

*/

public void post(Object event, String tag) {

if (event == null) {

Log.e(this.getClass().getSimpleName(), "The event object is null");

return;

}

mLocalEvents.get().offer(new EventType(event.getClass(), tag));

mDispatcher.dispatchEvents(event);

}

/**

* 发布Sticky事件,tag为EventType.DEFAULT_TAG

*

* @param event

*/

public void postSticky(Object event) {

postSticky(event, EventType.DEFAULT_TAG);

}

/**

* 发布含有tag的Sticky事件

*

* @param event 事件

* @param tag 事件tag

*/

public void postSticky(Object event, String tag) {

if (event == null) {

Log.e(this.getClass().getSimpleName(), "The event object is null");

return;

}

EventType eventType = new EventType(event.getClass(), tag);

eventType.event = event;

mStickyEvents.add(eventType);

}

public void removeStickyEvent(Class> eventClass) {

removeStickyEvent(eventClass, EventType.DEFAULT_TAG);

}

/**

* 移除Sticky事件

*

* @param type

*/

public void removeStickyEvent(Class> eventClass, String tag) {

Iterator iterator = mStickyEvents.iterator();

while (iterator.hasNext()) {

EventType eventType = iterator.next();

if (eventType.paramClass.equals(eventClass)

&& eventType.tag.equals(tag)) {

iterator.remove();

}

}

}

public List getStickyEvents() {

return mStickyEvents;

}

/**

* 设置订阅函数匹配策略

*

* @param policy 匹配策略

*/

public void setMatchPolicy(MatchPolicy policy) {

mDispatcher.mMatchPolicy = policy;

}

/**

* 设置执行在UI线程的事件处理器

*

* @param handler

*/

public void setUIThreadEventHandler(EventHandler handler) {

mDispatcher.mUIThreadEventHandler = handler;

}

/**

* 设置执行在post线程的事件处理器

*

* @param handler

*/

public void setPostThreadHandler(EventHandler handler) {

mDispatcher.mPostThreadHandler = handler;

}

/**

* 设置执行在异步线程的事件处理器

*

* @param handler

*/

public void setAsyncEventHandler(EventHandler handler) {

mDispatcher.mAsyncEventHandler = handler;

}

/**

* 返回订阅map

*

* @return

*/

public Map> getSubscriberMap() {

return mSubcriberMap;

}

/**

* 获取等待处理的事件队列

*

* @return

*/

public Queue getEventQueue() {

return mLocalEvents.get();

}

/**

* clear the events and subcribers map

*/

public synchronized void clear() {

mLocalEvents.get().clear();

mSubcriberMap.clear();

}

/**

* get the descriptor of EventBus

*

* @return the descriptor of EventBus

*/

public String getDescriptor() {

return mDesc;

}

public EventDispatcher getDispatcher() {

return mDispatcher;

}

/**

* 事件分发器

*

* @author mrsimple

*/

private class EventDispatcher {

/**

* 将接收方法执行在UI线程

*/

EventHandler mUIThreadEventHandler = new UIThreadEventHandler();

/**

* 哪个线程执行的post,接收方法就执行在哪个线程

*/

EventHandler mPostThreadHandler = new DefaultEventHandler();

/**

* 异步线程中执行订阅方法

*/

EventHandler mAsyncEventHandler = new AsyncEventHandler();

/**

* 缓存一个事件类型对应的可EventType列表

*/

private Map> mCacheEventTypes = new ConcurrentHashMap>();

/**

* 事件匹配策略,根据策略来查找对应的EventType集合

*/

MatchPolicy mMatchPolicy = new DefaultMatchPolicy();

/**

* @param event

*/

void dispatchEvents(Object aEvent) {

Queue eventsQueue = mLocalEvents.get();

while (eventsQueue.size() > 0) {

deliveryEvent(eventsQueue.poll(), aEvent);

}

}

/**

* 根据aEvent查找到所有匹配的集合,然后处理事件

*

* @param type

* @param aEvent

*/

private void deliveryEvent(EventType type, Object aEvent) {

// 如果有缓存则直接从缓存中取

List eventTypes = getMatchedEventTypes(type, aEvent);

// 迭代所有匹配的事件并且分发给订阅者

for (EventType eventType : eventTypes) {

handleEvent(eventType, aEvent);

}

}

/**

* 处理单个事件

*

* @param eventType

* @param aEvent

*/

private void handleEvent(EventType eventType, Object aEvent) {

List subscriptions = mSubcriberMap.get(eventType);

if (subscriptions == null) {

return;

}

for (Subscription subscription : subscriptions) {

final ThreadMode mode = subscription.threadMode;

EventHandler eventHandler = getEventHandler(mode);

// 处理事件

eventHandler.handleEvent(subscription, aEvent);

}

}

private List getMatchedEventTypes(EventType type, Object aEvent) {

List eventTypes = null;

// 如果有缓存则直接从缓存中取

if (mCacheEventTypes.containsKey(type)) {

eventTypes = mCacheEventTypes.get(type);

} else {

eventTypes = mMatchPolicy.findMatchEventTypes(type, aEvent);

mCacheEventTypes.put(type, eventTypes);

}

return eventTypes != null ? eventTypes : new ArrayList();

}

void dispatchStickyEvents(Object subscriber) {

for (EventType eventType : mStickyEvents) {

handleStickyEvent(eventType, subscriber);

}

}

/**

* 处理单个Sticky事件

*

* @param eventType

* @param aEvent

*/

private void handleStickyEvent(EventType eventType, Object subscriber) {

List eventTypes = getMatchedEventTypes(eventType, eventType.event);

// 事件

Object event = eventType.event;

for (EventType foundEventType : eventTypes) {

Log.e("", "### 找到的类型 : " + foundEventType.paramClass.getSimpleName()

+ ", event class : " + event.getClass().getSimpleName());

final List subscriptions = mSubcriberMap.get(foundEventType);

if (subscriptions == null) {

continue;

}

for (Subscription subItem : subscriptions) {

final ThreadMode mode = subItem.threadMode;

EventHandler eventHandler = getEventHandler(mode);

// 如果订阅者为空,那么该sticky事件分发给所有订阅者.否则只分发给该订阅者

if (isTarget(subItem, subscriber)

&& (subItem.eventType.equals(foundEventType)

|| subItem.eventType.paramClass

.isAssignableFrom(foundEventType.paramClass))) {

// 处理事件

eventHandler.handleEvent(subItem, event);

}

}

}

}

/**

* 如果传递进来的订阅者不为空,那么该Sticky事件只传递给该订阅者(注册时),否则所有订阅者都传递(发布时).

*

* @param item

* @param subscriber

* @return

*/

private boolean isTarget(Subscription item, Object subscriber) {

Object cacheObject = item.subscriber != null ? item.subscriber.get() : null;

return subscriber == null || (subscriber != null

&& cacheObject != null && cacheObject.equals(subscriber));

}

private EventHandler getEventHandler(ThreadMode mode) {

if (mode == ThreadMode.ASYNC) {

return mAsyncEventHandler;

}

if (mode == ThreadMode.POST) {

return mPostThreadHandler;

}

return mUIThreadEventHandler;

}

} // end of EventDispatcher

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值