基于 RxJava 的事件总线

1、前言

在MVP架构中,view层通常会像presenter层发送事件来告知view层需要更新。这就带来一个问题:view和presenter之前互相持有,没有解决耦合问题。如何设计一种方式来解决这个问题呢?

2、RxJava

其实仔细想想,view层和presenter层直接的事件完全可以用观察者模式来实现,那RxJava应该能解决这个问题。有关RxJava可以查看官网这个博客

3、实现

3.1 建立全局单例的RxBus类,给出实现

public final class RxBus {
    public static RxBus getInstance() {
        return Holder.BUS;
    }

    private static class Holder {
        private static final RxBus BUS = new RxBus();
    }

    private final Subject<Object> mBus;

    private RxBus() {
        mBus = PublishSubject.create().toSerialized();
        mStickyEvents = new ConcurrentHashMap<>();
    }

    public void post(Object obj) {
        mBus.onNext(obj);
    }

    public <T> Observable<T> toObservable(Class<T> clazz) {
        return mBus.ofType(clazz);
    }

    public Observable<Object> toObservable() {
        return mBus.hide();
    }

    public boolean hasObservers() {
        return mBus.hasObservers();
    }

3.2 在view层发送事件

    private void postDelayed(IControlEvent IControlEvent) {
        mPresenter.runPostDelayed(new Runnable() {
            @Override
            public void run() {
                RxBus.getInstance().post(mainControlEvent);
            }
        });
    }

这里mainControlEvent是IControlEvent的某个具体实现类。

3.3 presenter层接收事件

	private Disposable mSubscribe;
	
    protected void onSubscription() {
        super.onSubscription();
        RxUtils.dispose(mSubscribe);
        mSubscribe = RxBus.getInstance().toObservable(MainControlEvent.class)
                .distinctUntilChanged()
                .compose(applySchedulers())
                .subscribe(new Consumer<MainControlEvent>() {
                    @Override
                    public void accept(MainControlEvent controlEvent) {
                         //业务逻辑
                    }
                });

重要过程说明:
1、RxBus.getInstance()toObservable()

RxBus.getInstance().toObservable(MainControlEvent.class)

RxBus里,toObservable()方法的定义

public <T> Observable<T> toObservable(Class<T> clazz) {
        return mBus.ofType(clazz);
    }

根据RxJava的文档ofType的作用就是过滤出指定的类,这里是MainControlEvent这个类
在这里插入图片描述
2、.distinctUntilChanged()
根据API文档描述,这个方法的作用是滤除相同的对象:
在这里插入图片描述
3、.compose()
compose()方法的作用是将被观察者转换成指定的元素,这里

.compose(applySchedulers())

的作用是实现了线程间切换,并指定了去注册的时机:

	private ObservableTransformer mObservableTransformer;
    protected final <T> ObservableTransformer<T, T> applySchedulers() {
        if (mObservableTransformer == null) {
            mObservableTransformer = new ObservableTransformer<T, T>() {
                @NonNull
                @Override
                public ObservableSource<T> apply(@NonNull io.reactivex.Observable<T> upstream) {

                        if (mBaselovView.getActivity() != null && mBaselovView.lifecycle != null) {
                            return upstream.takeUntil(getDisposeSubject())
                                    .subscribeOn(Schedulers.io())
                                    .observeOn(AndroidSchedulers.mainThread())
                                    .compose(new SimpleLogTransformer<T>())
                                    .compose(mBaselovView.lifecycle.bindUntilEvent(ActivityEvent.DESTROY));
                        }
                    }

                    return upstream.takeUntil(getDisposeSubject())
                            .subscribeOn(Schedulers.io())
                            .compose(new SimpleLogTransformer<T>())
                            .observeOn(AndroidSchedulers.mainThread());
                }
            };
        }
        return (ObservableTransformer<T, T>) mObservableTransformer;
    }

在RxJava里,ObservableTransformer接口的定义就是将输入流转换为指定的输出流,当然也可以指定从io线程转换到android主线程。
4、.subscribe()
在.subscribe()方法里实现view层的更新

	.subscribe(new Consumer<MainControlEvent>() {
		@Override
		public void accept(MainControlEvent controlEvent) {
			if (controlEvent != null) {
                 //业务逻辑
			}
		}
	});

4、总结

以上,基于RxJava的MVP架构下view层到presenter层的事件以及presenter层到view层之间的更新就实现了。采用这种方法,view层和presenter层之间的解耦比较好,只是在代码中存在大量view和presenter的时候代码可能会难于追踪。

5、彩蛋

在以前的博文中,曾经分享过《Rxjava+retrofit框架备忘》。里面有RxJava和retrofit实现的请求框架,网络请求接口通过回调传递给UI。在今天介绍的这个基于RxJava的事件总线也可以和retrofit结合,实现另外一种风格的网络请求框架。

    public void sendControlComm(ControlEnum controlEnum, IControlEvent controlEvent) {
        Observable<BaseResponse<String>> responseObservable = null;
        
		responseObservable = ControlRepository.getInstance().operationEngine(
                        new OperationRequest(controlEnum == ControlEnum.ENGINE_START_OPEN
                                ? ControlCons.STATUS.ON : ControlCons.STATUS.OFF));
		……
    	responseObservable.subscribeOn(Schedulers.io())
                .compose(new SimpleLogTransformer<BaseResponse<String>>())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new ControlObserver(controlEvent)); 
        ……
	}

	protected class ControlObserver<Event extends IControlEvent> extends SimpleObserver<BaseResponse<String>> {

        private Event mControlEvent;

        public ControlObserver(Event controlEvent) {
            this.mControlEvent = controlEvent;
        }

        @Override
        public void onNext(BaseResponse<String> response) {
            super.onNext(response);
				……
                RxBus.getInstance().post(response.getData());
        }

        @Override
        public void onError(Throwable e) {
            super.onError(e);
			……
        }
    }                  

采用这种方法,同样也实现了RxJava+retrofit的网络请求框架。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值