Rx实现事件总线(类似于EventBus)总结
2017-01-06 11:47
417 查看
前言
这里使用的是Rx实现的事件总线,类似于EventBus和Otto,使用rx实现,有效的减少jar大小,并且使用和EventBus差不多,也是发送事件,然后被观察者订阅事件并在接收到事件之后做响应的处理。例子(用在这里不是很好(毕竟就两个activity,更好的可以通过回调))
这里使用的是登录,如果登录成功,就在LoginActivity里面发送一个UserEvent事件,然后在MAinActivity里面进行注册事件,并对注册之后接收到的事件进行处理。LoginActivity.class里面逻辑
mDataBean = user.getData(); if (mDataBean != null){ String username = mDataBean.getUsername(); String imageUrl = mDataBean.getLogo_url(); String useremail = mDataBean.getEmail(); UserEvent event = new UserEvent(username,imageUrl,useremail); RxBus.getDefault().post(event); new Handler().postDelayed(new Runnable() { @Override public void run() { mBuilder.build().dismiss(); LoginActivity.this.finish(); } },500); }
MainActivity.class里面逻辑
() { @Override public UserEvent call(UserEvent userEvent) { return userEvent; } }) .subscribe(new RxBusSubscriber() { @Override public void onEvent(UserEvent userEvent) { if (userEvent != null){ mTvEmail.setText(userEvent.mUserEmail); mTvName.setText(userEvent.mUserName); Glide.with(MainActivity.this) .load(userEvent.mImageUrl).asBitmap() .into(mIvAvator); } } @Override public void onError(Throwable e) { super.onError(e); } }); } " data-snippet-id="ext.4fd500f41d439f56ecf6b9cc486a6e06" data-snippet-saved="false" data-codota-status="done">[code] /** * 接受到事件并做相关处理 */ private void subscribeEvent(){ RxSubscriptions.remove(mRxSub); mRxSub = RxBus.getDefault().toObservable(UserEvent.class) .map(new Func1<UserEvent, UserEvent>() { @Override public UserEvent call(UserEvent userEvent) { return userEvent; } }) .subscribe(new RxBusSubscriber<UserEvent>() { @Override public void onEvent(UserEvent userEvent) { if (userEvent != null){ mTvEmail.setText(userEvent.mUserEmail); mTvName.setText(userEvent.mUserName); Glide.with(MainActivity.this) .load(userEvent.mImageUrl).asBitmap() .into(mIvAvator); } } @Override public void onError(Throwable e) { super.onError(e); } }); }
当然了,不要忘记注册事件监听
/*注册订阅RxBus事件*/ subscribeEvent();
这个时候就可以实现,当登录成功之后返回,用户名和头像等详情信息就会更新.其实这个逻辑还是很适合的,毕竟如果左侧有用户名和头像,而且主页的title的左侧也是一个头像的话,利用这个逻辑,还是很合适的。并且同一个被观察者可以被多个观察者订阅。逻辑代码是一样的,注册订阅同一个被观察者就行。
待续(粘性事件)
有的时候我们可能会在注册观察者之前就去发送了事件,但是如果我们直接发送了一般的事件,那么由于我们没有注册接收事件,这个时候我们即使后来注册了接收事件,那么也是接收不到的,这个时候我们就想到了粘性事件。粘性事件就好比我们在任何时候发送,发送之后如果没有注册者去接收,那么就会缓存到一个cache中,等待观察者接收,等到观察者注册之后接收并且去消费。当然了RxBus的粘性事件只会被消费最后一个被保存的事件。
我们先来看下如下的一个例子。
这里我们先发送一个粘性事件,当然例子当中为了对比,我们的一般事件和粘性事件都在点击的时候触发
mTest.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { RxBus.getDefault().post(new TestEvent("Normal Ceshi")); RxBus.getDefault().postSticky(new TestEvent("Sticky Ceshi")); startActivity(new Intent(MainActivity.this,TestActivity.class)); } });
当然了,虽然都触发,但是我们传递的消息是不通的(便于对比),这个时候我们在跳转之后的TestActivity里面进行观察者注册和接收。
我们看下逻辑写法。(当然一般的注册和以上的代码是一样的,这里就不写了,直接上粘性事件注册)
() { @Override public TestEvent call(TestEvent testEvent) { try { //在这里主动进行异常拦截 } catch (Exception e){ } return testEvent; } }).subscribe(new RxBusSubscriber() { @Override public void onEvent(TestEvent testEvent) { Toast.makeText(TestActivity.this, testEvent.test, Toast.LENGTH_SHORT).show(); } @Override public void onError(Throwable e) { super.onError(e); } }); } RxSubscriptions.add(mRxStickySub); }" data-snippet-id="ext.f9867b42b501de80cc90f8b81d2e6167" data-snippet-saved="false" data-codota-status="done">[code] /** * 接受粘性事件并做相关处理 */ public void subscribeStickyEvent(){ if (mRxStickySub != null && !mRxStickySub.isUnsubscribed()){ RxSubscriptions.remove(mRxStickySub); //移除 } else { TestEvent testEvent = RxBus.getDefault().getStickyEvent(TestEvent.class); mRxStickySub = RxBus.getDefault().toObservableSticky(TestEvent.class) .map(new Func1<TestEvent, TestEvent>() { @Override public TestEvent call(TestEvent testEvent) { try { //在这里主动进行异常拦截 } catch (Exception e){ } return testEvent; } }).subscribe(new RxBusSubscriber<TestEvent>() { @Override public void onEvent(TestEvent testEvent) { Toast.makeText(TestActivity.this, testEvent.test, Toast.LENGTH_SHORT).show(); } @Override public void onError(Throwable e) { super.onError(e); } }); } RxSubscriptions.add(mRxStickySub); }
我们来看下效果图吧。
注意
一定要在activity或者fragment销毁的时候调用@Override protected void onDestroy() { super.onDestroy(); //这里进行移除 也是可以测试正常的发送事件如果在注册之前发送 是接收不到的 RxSubscriptions.remove(mRxSub); //而这个就是可以的,及时事件发送是在注册之前,但是粘性事件会保留,直到有观察者注册事件并且去接收 RxSubscriptions.remove(mRxStickySub); }
因为在之前我没有去调用这个销毁(防止内存泄漏),当我在此点击跳转的时候,一般事件也出现了弹框,而且粘性事件出现了3次(~~(>_<)~~)。
基本上以上就可以满足一般的项目用法。
代码传送门
https://github.com/wuyinlei/RxBus感谢作者
感谢@YoKey,有兴趣的可以看下http://www.jianshu.com/p/ca090f6e2fe2再次谢谢作者,这里我只是学习记录。方便于自己以后查找。
相关文章推荐
- C#总结(六)EventBus事件总线的使用-自己实现事件总线
- EventBus使用详解-2-用RxJava实现事件总线(Event Bus)
- 借助事件总线框架(EventBus)实现Fragment与Activity通信
- 关于事件总线框架EventBus和otto一些分析总结
- AndroidEventBus ( 事件总线 ) 的设计与实现
- Android 框架学习2:源码分析 EventBus 3.0 如何实现事件总线
- AndroidEventBus ( 事件总线 ) 的设计与实现
- EventBus 事件总线的实现原理
- 用RxJava实现Rxbus替换EventBus事件总线
- android事件总线(eventbus)设计与实现
- 教你自己实现一个事件总线EventBus
- Android开发之Button事件实现方法的总结
- Android学习之Button事件实现方法的总结
- AndroidEventBus,android事件总线框架
- 使用事件总线框架EventBus和Otto
- AndroidEventBus事件总线的使用
- Android学习系列(43)--使用事件总线框架EventBus和Otto
- 使用事件总线框架EventBus和Otto
- Android事件总线 ( AndroidEventBus ) 开源库发布
- Guava(事件总线): 事件总线EventBus