简单的实现方式
- 来源的列子:(http://www.jianshu.com/p/ca090f6e2fe2)
看代码:
public class RxBus { private static volatile RxBus defaultInstance; private final Subject<Object, Object> bus; // PublishSubject只会把在订阅发生的时间点之后来自原始Observable的数据发射给观察者 public RxBus() { bus = new SerializedSubject<>(PublishSubject.create()); } // 单例RxBus public static RxBus getDefault() { if (defaultInstance == null) { synchronized (RxBus.class) { if (defaultInstance == null) { defaultInstance = new RxBus(); } } } return defaultInstance ; } // 发送一个新的事件 public void post (Object o) { bus.onNext(o); } // 根据传递的 eventType 类型返回特定类型(eventType)的 被观察者 public <T> Observable<T> toObservable (Class<T> eventType) { return bus.ofType(eventType); // 这里感谢小鄧子的提醒: ofType = filter + cast // return bus.filter(new Func1<Object, Boolean>() { // @Override // public Boolean call(Object o) { // return eventType.isInstance(o); // } // }) .cast(eventType); } }
Subject是非线程安全,在RxBus的使用场景中,存在并发情况,SerializedSubject是线程安全的,官方推荐用SerializedSubject封装Subject实现线程安全(SerializedSubject是Subject的子类)
事件发送代码
RxBus.getDefault().post(new UserEvent (1, "yoyo"));
事件接收代码
// rxSubscription是一个Subscription的全局变量,这段代码可以在onCreate/onStart等生命周期内 rxSubscription = RxBus.getDefault().toObserverable(UserEvent.class) .subscribe(new Action1<UserEvent>() { @Override public void call(UserEvent userEvent) { long id = userEvent.getId(); String name = userEvent.getName(); ... } }, new Action1<Throwable>() { @Override public void call(Throwable throwable) { // TODO: 处理异常 } });
- 上面例子仅仅说明实现的基本原理,存在的问题(或者说无法满足实际的应用)
- 没有sticky事件(Sticky事件只指事件消费者在事件发布之后才注册的也能接收到该事件的特殊类型)
- 你会发现在你订阅了某个事件后,在后续接收到该事件时,处理的过程中发生了异常,你可能会发现后续的事件都接收不到了!