EventBus事件总线详情

一, GreenRobot EventBus

1 simple demo
  • 定义event bus
package com.example.demo.common.config;

import org.greenrobot.eventbus.EventBus;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

/**
 * @author leiming
 * 2021/1/26 10:25
 * @description 事件总线配置
 */
@Configuration
public class EventBusConfig {

    @Bean
    public EventBus eventBus() {
        return new EventBus();
    }

}
  • 封装事件传递消息
/**
 * @author leiming
 * 2021/1/26 11:27
 * @description 事件消息
 */
@Data
@Builder
public class EventMessage implements Serializable {

    private String msg;
}
  • 定义事件监听
package com.example.demo.event;

import lombok.extern.slf4j.Slf4j;
import org.greenrobot.eventbus.Subscribe;
import org.springframework.stereotype.Component;

/**
 * @author leiming
 * 2021/1/26 10:30
 * @description 事件监听器
 */
@Component
@Slf4j
public class EventListener {

    @Subscribe
    public void onMessage(EventMessage eventMsg) {
        log.info("subscribe message :{}", eventMsg.getMsg());
    }
}
  • 注册event bus
package com.example.demo.event;

import lombok.extern.slf4j.Slf4j;
import org.greenrobot.eventbus.EventBus;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;

/**
 * @author leiming
 * 2021/1/26 10:32
 * @description 事件处理
 */
@Component
@Slf4j
public class EventHandler {

    @Autowired
    private EventBus eventBus;

    @Autowired
    private EventListener eventListener;
	//初始化注册事件subscriber
    @PostConstruct
    public void init() {
        eventBus.register(eventListener);
    }
	//销毁事件subscriber
    @PreDestroy
    public void destroy() {
        eventBus.unregister(eventListener);
    }
	//向event bus传递消息
    public void eventPost() {
        eventBus.post(EventMessage.builder().msg("hello event").build());
        log.info("post event msg");
    }
}
//如果event bus post的消息类型和subscriber定义接收的消息类型不一致,则subscriber不会接收到消息
  • 调用EventHandler.eventPost()
11:37:55.817INFO  com.example.demo.event.EventListener - subscribe message :hello event
11:37:56.028INFO  com.example.demo.event.EventHandler - post event msg
2 详解
  • @Subscribe
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
public @interface Subscribe {
    ThreadMode threadMode() default ThreadMode.POSTING;

    /**
     * If true, delivers the most recent sticky event (posted with
     * {@link EventBus#postSticky(Object)}) to this subscriber (if event available).
     */
    boolean sticky() default false;

    /** Subscriber priority to influence the order of event delivery.
     * Within the same delivery thread ({@link ThreadMode}), higher priority subscribers will receive events before
     * others with a lower priority. The default priority is 0. Note: the priority does *NOT* affect the order of
     * delivery among subscribers with different {@link ThreadMode}s! */
    int priority() default 0;
}

- ThreadMode.POSTING:subscriber的调用和post处于同一线程,默认选项。避免了线程的切换,开销最小。在该模式下subscriber处理过程需要简单耗时小,避免阻塞post线程。

- ThreadMode.MAIN:在Android中,如果post线程是主线程(ui线程),subscriber将在同一线程中被调用;否则,subscriber将以非阻塞的方式排队等待调用。如果不是在Android中,行为和上述POSTING一致。

- ThreadMode.MAIN_ORDERED:和MAIN不同的地方在于,该模式总是排队等待调用。

- ThreadMode.BACKGROUND:在Android中subscriber将会在后台进程中调用;如果post线程不是main线程,则subscriber直接在post线程中调用;如果post线程是main线程,则subscriber用一个独立的后台进程调用。如果不在Android中则subscriber总是使用后台线程进行调用。

- ThreadMode.ASYNC:subscriber会在一个独立的线程中异步调用,独立于main线程和post线程。envent bus使用线程池控制线程并发数量。

  • sticky事件

sticky事件:粘性事件,该类型事件在发送后,除了当前注册的subscriber可以监听到消息外,之后注册的subscriber也同样可以接收到先前发送的sticky事件。在一些场景中可以利用该特性实现缓存的动态刷新,历史信息加载等。

EventBus keeps the last sticky event of a certain type in memory.

  • priority

priority定义一个订阅者的优先级,在同一个ThreadMode中,优先级高的会首先接收到事件信息;在不同的ThreadMode中互不影响。

在一个方法上添加注解@Subscribe即可声明为事件的订阅者,@Subscribe注解中的属性sticky设置为true方可接收注册前的sticky事件。

/**
 * @author leiming
 * 2021/1/27 15:12
 * @description 粘性事件监听器
 */
@Component
@Slf4j
public class StickyEventListener {

    @Subscribe(sticky = true)
    public void listening(EventMessage msg) {
        log.info("sticky listener get msg:{}", msg.getMsg());
    }
}
@Autowired
private StickyEventListener stickyEventListener;

public void eventPost() {
    eventBus.postSticky(EventMessage.builder().msg("sticky event msg").build());
    log.info("finish post event msg");
    eventBus.register(stickyEventListener);
}
//调用该方法,可见当sticky事件发送后再注册stickyEventListener,仍然可以接收到该粘性事件 
15:41:54.357INFO  com.example.demo.event.EventHandler - finish post event msg
15:41:54.357INFO  com.example.demo.event.StickyEventListener - sticky listener get msg:sticky event msg

– 注意:subscriber方法的参数必须和post的事件信息的类型相同,否则订阅者(subscriber)无法接收事件信息。

  • EventBus定义
//1.使用默认配置的eventBus
EventBus eventBus = EventBus.getDefault();
//2.手动创建
EventBus eventBus = new EventBus();
//3.自定义,more see {org.greenrobot.eventbus.EventBusBuilder}
EventBus eventBus = EventBus.builder()
    //当subscriber抛出异常时打印日志,false时不打印
    .logSubscriberExceptions(false)
    //当没有subscriber时打印日志
    .logNoSubscriberMessages(false)
    //没有订阅者时是否发送事件
    .sendNoSubscriberEvent(false)
    .sendSubscriberExceptionEvent(true)
    //当订阅者出现异常时,event bus抛出该异常,并终止执行
    .throwSubscriberException(false)
    .build();
  • 取消event传递

eventBus.cancelEventDelivery(eventMsg):可以在一个subscriber中手动调用该方法,取消一个event的传递,对于同一个event的阻塞序列中的subscriber将不会接收到该event。

二, Guava EventBus

和GreenRobot EventBus类似