Bootstrap

EventBus:基本使用详解

一、简介

EventBus项目:https://github.com/greenrobot/EventBus

EventBus 3.0.0 API:http://greenrobot.org/files/eventbus/javadoc/3.0/

EventBus是一种用于Android/Java的事件发布-订阅总线框架。

img

特点:

  • 简化组件之间的通信

    • 分离事件发送者和接收者

    • 很好地处理Activities、Fragments和后台线程

    • 避免复杂且易出错的依赖关系和生命周期问题

  • 使代码更简单

  • 很快

  • 很小

  • 通过安装超过100000000次的应用程序在实践中得到验证

  • 具有高级功能,如传递线程、订阅服务器优先级等。

查看源码详解请看:《EventBus:源码详解》

二、导入库

Gradle:

implementation 'org.greenrobot:eventbus:3.1.1'

Maven:

<dependency>
    <groupId>org.greenrobot</groupId>
    <artifactId>eventbus</artifactId>
    <version>3.1.1</version>
</dependency>

通过Maven Central下载 最新 JAR

三、使用步骤

1. 定义事件

public static class MessageEvent { /* Additional fields if needed */ }

2. 准备订阅服务

声明并标注订阅方法,还可以指定线程模式:

@Subscribe(threadMode = ThreadMode.MAIN)  
public void onMessageEvent(MessageEvent event) {/* Do something */};

注册和注销订阅服务。例如在Android上,Activities、Fragments通常应该根据其生命周期进行注册:

 @Override
 public void onStart() {
     super.onStart();
     EventBus.getDefault().register(this);
 }

 @Override
 public void onStop() {
     super.onStop();
     EventBus.getDefault().unregister(this);
 }

3. 发布事件

 EventBus.getDefault().post(new MessageEvent());

4. 简单使用实例

1)构造事件(Event)对象。也就是发送消息类
每一个消息类,对应一种事件。这里我们定义两个消息发送类。后面讲解具体作用。

public class ToastEvent {
    private String content;

    public ToastEvent(String content) {
        this.content = content;
    }

    public String getContent() {
        return content;
    }

    public void setContent(String content) {
        this.content = content;
    }
}

2)注册/解除事件订阅(Subscriber)

EventBus.getDefault().register(this);//注册事件 其中this代表订阅者

具体注册了对什么事件的订阅,这个需要onEvent()方法来说明。在EventBus 3.0之前,onEvent()方法是用来接收指定事件(Event)类型对象,然后进行相关处理操作。在EventBus 3.0之后,onEvent()方法可以自定义方法名,不过要加入注解@Subscribe。

@Subscribe
    public void onToastEvent(ToastEvent event){
        Toast.makeText(MainActivity.this,event.getContent(),Toast.LENGTH_SHORT).show();
    }

通过register(this)来表示该订阅者进行了订阅,通过onToastEvent(ToastEvent event)表示指定对事件ToastEvent的订阅。到这里订阅就完成了。

需要注意的是:一般在onCreate()方法中进行注册订阅。在onDestory()方法中进行解除订阅。

@Override
    protected void onDestroy() {
        super.onDestroy();
        EventBus.getDefault().unregister(this);
    }

3)发送消息
订阅已经完成,那么便可以发送订阅了。

EventBus.getDefault().post(new ToastEvent("Toast,发个提示,祝大家新年快乐!"));

那么onToastEvent(ToastEvent event)会收到事件,并弹出提示。

EventBus的基础使用流程就是这样的。

四、使用详解

1. ThreadMode

EventBus可以处理线程:事件可以在不同于发布线程的线程中发布。

一个常见的例子是处理UI改变。在Android中,必须在主线程中进行UI更改,联网或任何耗时的任务都不得在主线程上运行。EventBus可以处理这些任务并与UI线程同步(而不必深入研究线程转换,使用AsyncTask等)。

在EventBus中,可以使用下列ThreadModes之一来定义将调用事件处理方法的线程。

ThreadMode:POSTING

订阅者将在发布事件的同一线程中被调用。这是默认值。事件传递是同步完成的,发布完成后,所有订阅者都将被呼叫。

此ThreadMode意味着开销最少,因为它可以完全避免线程切换。因此,这是简单任务的推荐模式,该任务在很短的时间内即可完成,而无需主线程。

使用此模式的事件处理程序应快速返回,以避免阻塞可能是主线程的发布线程。

例:

// Called in the same thread (default)
// ThreadMode is optional here
@Subscribe(threadMode = ThreadMode.POSTING)
public void onMessage(MessageEvent event) {
    log(event.message);
}
ThreadMode:MAIN

订阅者将在Android的主线程中调用。如果发布线程是主线程,则将直接调用事件处理程序方法。

使用此模式的事件处理程序必须快速返回以避免阻塞主线程。

例:

// Called in Android UI's main thread
@Subscribe(threadMode = ThreadMode.MAIN)
public void onMessage(MessageEvent event) {
    textField.setText(event.message);
}
ThreadMode:MAIN_ORDERED

订阅者将在Android的主线程中调用。该事件始终排入队列,通过队列传递给订阅者。事件处理具有更严格和更一致的顺序(因此名称为MAIN_ORDERED)。

例如,如果在具有MAIN线程模式的事件处理程序中发布另一个事件,则第二个事件处理程序将在第一个事件处理程序之前完成(因为它被同步调用:将它与方法调用进行比较)。使用MAIN_ORDERED,第一个事件处理程序将完成,然后第二个事件处理程序将在稍后的时间点(主线程具有容量后)被调用。

使用此模式的事件处理程序必须快速返回以避免阻塞主线程。

例:

// Called in Android UI's main thread
@Subscribe(threadMode = ThreadMode.MAIN_ORDERED)
public void onMessage(MessageEvent event) {
    textField.setText(event.message);
}
ThreadMode:BACKGROUND

订阅者将在后台线程中被调用。

如果发布线程不是主线程,则将在发布线程中直接调用事件处理程序方法。如果发布线程是主线程,则EventBus使用单个后台线程,该线程将顺序传递其所有事件。

使用此模式的事件处理程序应尝试快速返回以避免阻塞后台线程。

例:

// Called in the background thread
@Subscribe(threadMode = ThreadMode.BACKGROUND)
public void onMessage(MessageEvent event){
    saveToDisk(event.message);
}
ThreadMode:ASYNC

事件处理程序方法在单独的线程中调用。这始终独立于发布线程和主线程。

发布事件永远不会等待使用此模式的事件处理程序方法。

如果事件处理程序方法的执行可能需要一些时间(例如,用于网络访问),则应使用此模式。避免同时触发大量长时间运行的异步处理程序方法,以限制并发线程数。

EventBus使用线程池来有效地重用已完成的异步事件处理程序通知中的线程。

例:

// Called in a separate thread
@Subscribe(threadMode = ThreadMode.ASYNC)
public void onMessage(MessageEvent event){
    backend.send(event.message);
}
总结
  • ThreadMode.POSTING:默认的模式,开销最小的模式,因为声明为POSTING的订阅者会在发布的同一个线程调用,发布者在主线程那么订阅者也就在主线程,反之亦,避免了线程切换,如果不确定是否有耗时操作,谨慎使用,因为可能是在主线程发布

  • ThreadMode.MAIN:主线程调用,视发布线程不同处理不同,如果发布者在主线程那么直接调用(非阻塞式),如果发布者不在主线程那么阻塞式调用

  • ThreadMode.MAIN_ORDERED:和MAIN差不多,主线程调用,和MAIN不同的是他保证了post是非阻塞式的(默认走MAIN的非主线程的逻辑,所以可以做到非阻塞)

  • ThreadMode.BACKGROUND:在子线程调用,如果发布在子线程那么直接在发布线程调用,如果发布在主线程那么将开启一个子线程来调用,这个子线程是阻塞式的,按顺序交付所有事件,所以也不适合做耗时任务,因为多个事件共用这一个后台线程

  • ThreadMode.ASYNC: 在子线程调用,总是开启一个新的线程来调用,适用于做耗时任务,比如数据库操作,网络请求等,不适合做计算任务,会导致开启大量线程

2. 配置

EventBusBuilder类配置EventBus的各个方面。

例如,以下是构建EventBus的方法,该事件总线在发布的事件没有订阅者的情况下保持安静:

EventBus eventBus = EventBus.builder()
    .logNoSubscriberMessages(false)
    .sendNoSubscriberEvent(false)
    .build();

示例,当订阅者引发异常时失败:

EventBus eventBus = EventBus.builder().throwSubscriberException(true).build();

注意:默认情况下,EventBus捕获从订阅者方法引发的异常,并发送一个 SubscriberExceptionEvent,该事件可能要处理。

检查EventBusBuilder类及其JavaDoc,以获取所有可能的配置可能性。

配置默认的EventBus实例

使用EventBus.getDefault()是从应用程序中任何位置获取共享EventBus实例的简单方法。EventBusBuilder还允许使用**installDefaultEventBus ()**方法配置此默认实例 。

例如,可以将默认的EventBus实例配置为重新抛出在订阅者方法中发生的异常。但是,仅在DEBUG构建中使用此方法,因为这可能会在出现异常时使应用程序崩溃:

EventBus.builder().throwSubscriberException(BuildConfig.DEBUG).installDefaultEventBus();

注意:只能在第一次使用默认EventBus实例之前完成一次。随后对installDefaultEventBus()的调用将引发异常。这样可以确保您的应用程序中行为一致。Application类是在使用之前配置默认EventBus实例的好地方。

3. 粘性事件

在发布事件后,某些事件会携带需要的信息。例如,一个事件表示某些初始化已完成。或者,如果有一些传感器或位置数据,并且想要保留最新值。除了使用自己的缓存之外,还可以使用粘性事件。

因此,EventBus将某种类型的最后一个粘性事件保留在内存中。然后,粘性事件可以传递给订户或显式查询。因此,不需要任何特殊的逻辑即可考虑已经可用的数据。

粘性示例

假设某个粘性事件是在一段时间前发布的:

EventBus.getDefault().postSticky(new MessageEvent("Hello everyone!"));

现在开始一个新的Activity。在注册期间,所有粘性订阅者方法将立即获得先前发布的粘性事件

@Override
public void onStart() {
    super.onStart();
    EventBus.getDefault().register(this);
}
 
// UI updates must run on MainThread
@Subscribe(sticky = true, threadMode = ThreadMode.MAIN)
public void onEvent(MessageEvent event) {   
    textField.setText(event.message);
}
 
@Override
public void onStop() {
    EventBus.getDefault().unregister(this);    
    super.onStop();
}
手动获取和删除粘性事件

最后一个粘性事件在注册时自动传递给匹配的订阅者。但是有时手动检查粘性事件可能更方便。另外,可能有必要删除(使用)粘性事件,以便不再发送它们。例:

MessageEvent stickyEvent = EventBus.getDefault().getStickyEvent(MessageEvent.class);
// Better check that an event was actually posted before
if(stickyEvent != null) {
    // "Consume" the sticky event
    EventBus.getDefault().removeStickyEvent(stickyEvent);
    // Now do something with it
}

方法 removeStickyEvent被重载:当传入类时,它将返回之前保留的粘滞事件。使用此变体,可以改进前面的示例:

MessageEvent stickyEvent = EventBus.getDefault().removeStickyEvent(MessageEvent.class);
// Better check that an event was actually posted before
if(stickyEvent != null) {
    // Now do something with it
}

4. 优先事项和活动取消

大多数EventBus用例不需要优先级或事件取消,但在某些特殊情况下它们可能会派上用场。

订阅者优先级

可以通过在注册过程中为订阅者提供优先级来更改事件传递的顺序。

@Subscribe(priority = 1);
public void onEvent(MessageEvent event) {
    ...
}

在同一传递线程(ThreadMode)中,优先级较高的订户将在其他优先级较低的订户之前接收事件。默认优先级为0。

注意:优先级不影响具有不同ThreadModes的订户之间的传递顺序!

取消事件传送

可以通过从订阅者的事件处理方法中调用cancelEventDelivery (对象事件) 来取消事件传递过程 。任何进一步的事件传递将被取消,后续的订阅者将不会接收该事件。

// Called in the same thread (default)
@Subscribe
public void onEvent(MessageEvent event){
    // Process the event
    ...
    // Prevent delivery to other subscribers
    EventBus.getDefault().cancelEventDelivery(event) ;
}

事件通常由优先级较高的订阅者取消。取消仅限于在发布线程中运行的事件处理方法(ThreadMode.PostThread)。

5. 订阅者索引

订阅者索引是EventBus3的新功能。它是一项可选的优化,可加快初始订阅者注册的速度

订阅者索引可以在构建期间使用EventBus注释处理器创建。虽然不需要使用索引,但建议在Android上使用它以获得最佳性能

索引前提

请注意,只有@Subscriber方法可以被索引,并且订阅者和事件类是公共方法。同样,由于Java注释处理本身的技术限制,@ Subscribe注释无法在匿名类内部识别

当EventBus无法使用索引时,它将在运行时自动回退到反射。因此它仍然可以工作,只是速度稍慢一点。

生成索引

使用注解处理器

如果未使用2.2.0或更高版本的Android Gradle插件,请将该配置与android-apt配合使用。

要启用索引生成,需要使用annotationProcessor属性将EventBus注解处理器添加到build中。还设置一个参数 eventBusIndex以指定要生成的索引的完全限定的类。因此,例如,将以下部分添加到您的Gradle构建脚本中:

android {
    defaultConfig {
        javaCompileOptions {
            annotationProcessorOptions {
                arguments = [ eventBusIndex : 'com.example.myapp.MyEventBusIndex' ]
            }
        }
    }
}
 
dependencies {
    implementation 'org.greenrobot:eventbus:3.1.1'
    annotationProcessor 'org.greenrobot:eventbus-annotation-processor:3.1.1'
}

使用kapt

在Kotlin代码中使用EventBus,需要使用 kapt 代替 annotationProcessor :

apply plugin: 'kotlin-kapt' // ensure kapt plugin is applied
 
dependencies {
    implementation 'org.greenrobot:eventbus:3.1.1'
    kapt 'org.greenrobot:eventbus-annotation-processor:3.1.1'
}
 
kapt {
    arguments {
        arg('eventBusIndex', 'com.example.myapp.MyEventBusIndex')
    }
}

使用android-apt

如果以上方法都不适合,则可以使用android-apt Gradle插件将EventBus注释处理器添加到构建中。

将以下部分添加到您Gradle构建脚本中:

buildscript {
    dependencies {
        classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8'
    }
}
 
apply plugin: 'com.neenbedankt.android-apt'
 
dependencies {
    compile 'org.greenrobot:eventbus:3.1.1'
    apt 'org.greenrobot:eventbus-annotation-processor:3.1.1'
}
 
apt {
    arguments {
        eventBusIndex "com.example.myapp.MyEventBusIndex"
    }
}
使用索引

成功构建项目后,将为你生成用eventBusIndex指定的类 。然后在设置EventBus时将其传递为:

EventBus eventBus = EventBus.builder().addIndex(new MyEventBusIndex()).build();

或者,如果想在整个应用中使用默认实例:

EventBus.builder().addIndex(new MyEventBusIndex()).installDefaultEventBus();
// Now the default instance uses the given index. Use it like this:
EventBus eventBus = EventBus.getDefault();
索引你的库

可以将相同的原理应用于库(而不是最终应用程序)的一部分代码。可能具有多个索引类,可以在EventBus设置过程中全部添加它们,例如:

EventBus eventBus = EventBus.builder()
    .addIndex(new MyEventBusAppIndex())
    .addIndex(new MyEventBusLibIndex()).build();

6. 代码混淆

在开发中使用代码混淆的时候需要加上:

-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);
}

注意:无论是否使用订阅者索引都将需要此配置。

7. AsyncExecutor

AsyncExecutor就像一个线程池,但是具有异常处理。故障会引发异常,AsyncExecutor会将这些异常包装在事件中,该事件会自动发布。

声明:AsyncExecutor是一个非核心实用程序类。它可以为您节省一些在后台线程中进行错误处理的代码,但它不是EventBus的核心类。

通常,调用 AsyncExecutor.create ()创建实例并将其保留在Application范围内。然后执行一些操作,实现 RunnableEx接口并将其传递给AsyncExecutor的execute方法。与Runnable不同 , RunnableEx可能会引发Exception。

如果 RunnableEx实现引发异常,则将捕获该异常并将其包装到ThrowableFailureEvent中,并将其发布。

执行示例:

AsyncExecutor.create().execute(
    new AsyncExecutor.RunnableEx() {
        @Override
        public void run() throws LoginException {
            // No need to catch any Exception (here: LoginException)
            remote.login();
            EventBus.getDefault().postSticky(new LoggedInEvent());
        }
    }
);

接收部分的示例:

@Subscribe(threadMode = ThreadMode.MAIN)
public void handleLoginEvent(LoggedInEvent event) {
    // do something
}
 
@Subscribe(threadMode = ThreadMode.MAIN)
public void handleFailureEvent(ThrowableFailureEvent event) {
    // do something
}

AsyncExecutor生成器

如果要自定义AsyncExecutor实例,请调用静态方法 AsyncExecutor.builder()。它将返回一个生成器,可以自定义EventBus实例,线程池和failure事件的类

另一个定制选项是执行范围,它提供故障事件上下文信息。例如,失败事件可能仅与特定的Activity实例或类有关。

如果您的自定义失败事件类实现了HasExecutionScope接口,则AsyncExecutor将自动设置执行范围。这样,您的订户可以查询失败事件的执行范围,并据此作出反应。


我的学习笔记,欢迎star和fork

欢迎关注我的公众号,持续分析优质技术文章
欢迎关注我的公众号

;