EventBus是一个Android端优化的publish/subscribe消息总线,简化了应用程序内各组件间、组件与后台线程间的通信。比如请求网络,等网络返回时通过Handler或Broadcast通知UI,两个Fragment之间需要通过Listener通信,这些需求都可以通过EventBus实现
优点:简化组件之间的通信方式,实现解耦让业务代码更加简洁,可以动态设置事件处理线程以及优先级
缺点:目前发现唯一的缺点就是类似之前策略模式一样的诟病,每个事件都必须自定义一个事件类,造成事件类太多,无形中加大了维护成本
EventBus组成部分
Event 传递的事件对象
Subscriber 事件的订阅者
Publisher 事件的发布者
ThreadMode 定义函数在何种线程中执行
使用步骤
- 添加远程依赖
implementation 'org.greenrobot:eventbus:3.0.0'
- 设置混淆规则
-keepattributes *Annotation*
-keepclassmembers class ** {
@org.greenrobot.eventbus.Subscribe <methods>;
}
-keep enum org.greenrobot.eventbus.ThreadMode { *; }
# Only required if you use AsyncExecutor
-keepclassmembers class * extends org.greenrobot.eventbus.util.ThrowableFailureEvent {
<init>(java.lang.Throwable);
}
- Activity启动时注册,才能接收得到消息
EventBus.getDefault().register(MainActivity.this);
- Activity销毁时注销,才不会导致内存泄露
EventBus.getDefault().unregister(MainActivity.this);
- 创建事件类型
public class MessageEvent {
public String name;
public MessageEvent(String name) {
this.name = name;
}
}
- 接收事件
//订阅事件,并指定该事件在什么线程中执行
@Subscribe(threadMode = ThreadMode.MAIN)
public void MesssageEventBus(MessageEvent event){
// 显示接收的消息
textView.setText(event.name);
}
- 在其他类中发布事件
//这里发送的参数类型必须和接收的参数类型一致才能接收得到
EventBus.getDefault().post(new MessageEvent("HJQ"));
粘性事件
之前说的使用方法, 都是需要先注册(register),再post,才能接受到事件; 如果你使用postSticky发送事件, 那么可以不需要先注册, 也能接受到事件
创建事件类型
public class StickyEvent {
public String msg;
public StickyEvent(String msg) {
this.msg = msg;
}
}
- 发布粘性事件
EventBus.getDefault().postSticky(new StickyEvent("我是粘性事件"));
- 在其他类中接收粘性事件
//和之前的方法一样,只是多了一个 sticky = true 的属性.
@Subscribe(threadMode = ThreadMode.MAIN, sticky = true)
public void onEvent(StickyEvent event){
textView.setText(event.msg);
}
- 需要手动注册,注册后会自动接收并触发粘性事件
EventBus.getDefault().register(MainActivity.this);
- Activity销毁时注销,并移除粘性事件,才不会导致内存泄露
//移除所有粘性事件
EventBus.getDefault().removeAllStickyEvents();
//移除指定类型的粘性事件
//EventBus.getDefault().removeStickyEvent(new StickyEvent("我是粘性事件"));
EventBus.getDefault().unregister(MainActivity.this);
线程调度
@Subscribe(threadMode = ThreadMode.MAIN)
ThreadMode.MAIN:表示这个方法在主线程中执行
ThreadMode.BACKGROUND:表示该方法在后台执行,不能并发处理
ThreadMode.ASYNC:也表示在后台执行,可以异步并发处理
ThreadMode.POSTING:表示该方法和消息发送方在同一个线程中执行
设置优先级
@Subscribe(threadMode = ThreadMode.MAIN, priority = 100)
//终止事件往下传递,事件的优先级类似广播的优先级,优先级越高优先获得消息
EventBus.getDefault().cancelEventDelivery(event);