Spring注解 @EventListener 详解

基本概念


01

用途


将一个方法标记为监听器,用于监听应用程序事件,事件可以是ApplicationEvent实例,也可以是其他任意的对象。

如果一个监听器(被标注的方法)只支持单一的事件类型,那么该方法可以声明一个唯一的参数用来反映要监听的事件类型。

如果一个监听器(被标注的方法)支持多种事件类型,那么需要使用注解的classes属性指定一个或多个支持的事件类型。


02

事件处理条件

可以通过 condition 属性指定一个SpEL表达式,如果返回 “true”,“on”,“yes”, or“1” 中的任意一个,则事件会被处理,否则不会。


03

处理器


@EventListener注解的处理是通过内部的EventListenerMethodProcessorBean进行的。

当使用Java配置时,它被自动注册。

当使用XML配置时,则通过context:annotation-config/context:component-scan/元素手动注册。


04

返回值


被标注的方法可以没有返回值,也可以有返回值。

当有返回值是,其返回值会被当作为一个新的事件发送。如果返回类型是数组或集合,那么数组或集合中的每个元素都作为一个新的单独事件被发送。

注:只有同步监听器的返回值会被当成新的事件被发送。


05

异常处理


同步监听器抛出的所有checked异常都会被封装成UndeclaredThrowableException,因为事件发布者只能处理运行时异常(unchecked异常)。


06

异步监听器


当需要异步处理监听器时,可以在监听器方法上再增加另外的一个Spring注解@Async,但需要注意以下限制:

1.  监听器报错不会传递给事件发起者,因为双方已经不在同一个线程了。

2.  异步监听器的非空返回值不会被当作新的事件发布。如果需要发布新事件,需要注入ApplicationEventPublisher后手动发布。



07

监听器排序


如果同一个事件可能会被多个监听器监听处理,那么我们可以使用@Order注解对各个监听器进行排序。


08

源码


package org.springframework.context.event;

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import java.util.function.Predicate;

import org.springframework.context.ApplicationEvent;
import org.springframework.core.annotation.AliasFor;

@Target({ElementType.METHOD, ElementType.ANNOTATION_TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface EventListener {

  /**
   * 同 classes
   */

  @AliasFor("classes")
  Class<?>[] value() default {};

  /**
   * 可以处理的事件类型
   */

  @AliasFor("value")
  Class<?>[] classes() default {};

  /**
     * SpEL表达式判断是否满足处理条件
   */

  String condition() default "";

  /**
     * 可以给监听器指定一个id,默认是方法的全限定名,
     * 如:mypackage.MyClass.myMethod()
   */

  String id() default "";

}

Spring注解 @EventListener 详解


使用示例


01

单一事件监听器


发布事件

使用 ApplicationEventPublisher发布一个自定义事件 PersonSaveEvent


@Service
public class EventPublisher {

    private ApplicationEventPublisher eventPublisher;

    @Autowired
    public void setEventPublisher(ApplicationEventPublisher eventPublisher) {
        this.eventPublisher = eventPublisher;
    }

    public void publishPersonSaveEvent(){
        PersonSaveEvent saveEvent = new PersonSaveEvent();
        saveEvent.setId(1);
        saveEvent.setName("i余数");
        saveEvent.setAge(18);
        eventPublisher.publishEvent(saveEvent);
    }
}

监听事件

使用Spring注解@EventListener 标注方法为监听器。方法参数 PersonSaveEvent 为监听事件的类型。

@Slf4j
@Service
public class EventListenerService {

    @EventListener
    public void handleForPersonSaveEvent(PersonSaveEvent saveEvent){
        log.info("saveEvent -> {}", saveEvent);
    }
}

结果验证
事件监听成功,并执行了打印这个业务逻辑。

saveEvent -> PersonSaveEvent(id=1, name=i余数, age=18)


02

使用classes实现多事件监听器



发布事件
在上一个示例(
单一事件监听器)的基础上,再多加一个PersonUpdateEvent事件。

public void publishPersonUpdateEvent(){
    PersonUpdateEvent updateEvent = new PersonUpdateEvent();
    updateEvent.setId(1);
    updateEvent.setName("i余数");
    updateEvent.setAge(19);
    eventPublisher.publishEvent(updateEvent);
}


监听事件
使用classes属性指定多个被监听事件类型。

@EventListener(classes = {PersonSaveEvent.class, PersonUpdateEvent.class})
public void handleForPersonSaveAndUpdateEvent(Object event){
    log.info("multi handle event -> {}", event);
}


结果验证
多个事件监听成功,并执行了打印这个业务逻辑。

multi handle event -> PersonSaveEvent(id=1, name=i余数, age=18)
multi handle event -> PersonUpdateEvent(id=1, name=i余数, age=19)

03

使用condition筛选监听的事件


发布事件
发布两个编号分别 1 和 2 的 
PersonSaveEvent事件。

public void publishPersonSaveEvent(){
    PersonSaveEvent saveEvent = new PersonSaveEvent();
    saveEvent.setId(1);
    saveEvent.setName("i余数");
    saveEvent.setAge(18);
    eventPublisher.publishEvent(saveEvent);

    PersonSaveEvent saveEvent2 = new PersonSaveEvent();
    saveEvent2.setId(2);
    saveEvent2.setName("i余数");
    saveEvent2.setAge(18);
    eventPublisher.publishEvent(saveEvent2);
}


监听事件
使用 condition 属性指定只能监听编号为 1 的 PersonSaveEvent事件。

@EventListener(condition = "#root.event.getPayload().getId() == 1")
public void handleByCondition(PersonSaveEvent saveEvent){
    log.info("只处理id等于1的 -> {}", saveEvent);
}


结果验证
编号为 2 的事件不满足条件,所以不会执行。

只处理id等于1的 -> PersonSaveEvent(id=1, name=i余数, age=18)

04

有返回值的监听器


返回一个单一对象

发布事件

发布一个 PersonSaveEvent事件。


public void publishPersonSaveEvent(){
    PersonSaveEvent saveEvent = new PersonSaveEvent();
    saveEvent.setId(1);
    saveEvent.setName("i余数");
    saveEvent.setAge(18);
    eventPublisher.publishEvent(saveEvent);
}

监听事件
handleHaveReturn 监听 PersonSaveEvent事件,并返回一个 PersonUpdateEvent


handleForPersonUpdateEvent 监听 PersonUpdateEvent 事件。

@EventListener
public void handleForPersonUpdateEvent(PersonUpdateEvent updateEvent){
    log.info("handle update event -> {}", updateEvent);
}


@EventListener
public PersonUpdateEvent handleHaveReturn(PersonSaveEvent saveEvent){
    log.info("handle save event -> {}", saveEvent);
    PersonUpdateEvent updateEvent = new PersonUpdateEvent();
    updateEvent.setId(saveEvent.getId());
    updateEvent.setName(saveEvent.getName());
    updateEvent.setAge(saveEvent.getAge());
    return updateEvent;
}
结果验证

可以看到我们监听到了2个事件:

PersonSaveEvent是我们主动发布的事件。

PersonUpdateEvent 是 handleHaveReturn 方法的返回值,会被 Spring 自动当作一个事件被发送。


handle save event -> PersonSaveEvent(id=1, name=i余数, age=18)
handle update event -> PersonUpdateEvent(id=1, name=i余数, age=18)


返回一个集合

将监听器稍作修改,使其返回一个集合。

@EventListener
public List<PersonUpdateEvent> handleHaveReturn(PersonSaveEvent saveEvent){
    log.info("handle save event -> {}", saveEvent);
    List<PersonUpdateEvent> events = new ArrayList<>();
    PersonUpdateEvent updateEvent = new PersonUpdateEvent();
    updateEvent.setId(saveEvent.getId());
    updateEvent.setName(saveEvent.getName());
    updateEvent.setAge(saveEvent.getAge());
    events.add(updateEvent);

    PersonUpdateEvent updateEvent2 = new PersonUpdateEvent();
    BeanUtils.copyProperties(updateEvent, updateEvent2);
    events.add(updateEvent2);
    return events;

}

查看结果可以发现集合中的每个对象都被当作一个单独的事件进行发送

handle save event -> PersonSaveEvent(id=1, name=i余数, age=18)
handle update event -> PersonUpdateEvent(id=1, name=i余数, age=18)
handle update event -> PersonUpdateEvent(id=1, name=i余数, age=18)

返回数组和集合一样,数组中的每个对象都被当作一个单独的事件进行发送。


05

异步监听器


创建两个监听器,一个同步一个异步,异步监听器就是在方法上加一个 @Async 标签即可(实际使用中最好指定线程池)。


@EventListener
public void handleForPersonSaveEvent(PersonSaveEvent saveEvent){
    log.info("handle event -> {}", saveEvent);
}

@Async
@EventListener
public void handleForPersonSaveEventAsync(PersonSaveEvent saveEvent){
    log.info("async handle event -> {}", saveEvent);
}

事件发布和上面一样,就不做介绍了。

从执行结果可以看出,异步线程是 task-1,不是主线程 main,即异步是生效的。

INFO 3851 --- [ main] i.k.s.e.listener.EventListenerService : handle event -> PersonSaveEvent(id=1, name=i余数, age=18)
INFO 3851 --- [ task-1] i.k.s.e.listener.EventListenerService : async handle event -> PersonSaveEvent(id=1, name=i余数, age=18)

06

同步监听器异常处理


使用 SimpleApplicationEventMulticaster 
处理同步监听器抛出异常。

先定义一个
ErrorHandler:

@Slf4j
@Component
public class MyErrorHandler implements ErrorHandler {
    @Override
    public void handleError(Throwable t) {
        log.info("handle error -> {}", t.getClass());
    }
}

将自定义ErrorHandler绑定到
 SimpleApplicationEventMulticaster

@Slf4j
@Service
public class EventListenerService {

    @Autowired
    private SimpleApplicationEventMulticaster simpleApplicationEventMulticaster;

    @Autowired
    private MyErrorHandler errorHandler;

    @PostConstruct
    public void init(){
        simpleApplicationEventMulticaster.setErrorHandler(errorHandler);
    }

    @Order(1)
    @EventListener
    public void handleForPersonSaveEvent(PersonSaveEvent saveEvent) throws AuthException {
        log.info("handle event -> {}", saveEvent);
        throw new AuthException("test exception");
    }
}

结果可以看到捕获的异常是
 UndeclaredThrowableException

handle event -> PersonSaveEvent(id=1, name=i余数, age=18)
handle error -> class java.lang.reflect.UndeclaredThrowableException



07

异步监听器异常处理


使用 SimpleAsyncUncaughtExceptionHandler 

来处理 @Async 抛出的异常。


@Configuration
public class AsyncConfig implements AsyncConfigurer {
    @Override
    public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {
        return new SimpleAsyncUncaughtExceptionHandler();
    }
}

我们在异步监听器代码中人为的抛出一个异常。


@Async
@EventListener
public void handleForPersonSaveEvent(PersonSaveEvent saveEvent) throws AuthException {
    log.info("handle event -> {}", saveEvent);
    throw new AuthException("test exception");
}

可以看到
SimpleAsyncUncaughtExceptionHandler
捕获到了 @Async 方法抛出的异常。

INFO 4416 --- [ task-1] i.k.s.e.listener.EventListenerService : handle event -> PersonSaveEvent(id=1, name=i余数, age=18)
ERROR 4416 --- [ task-1] .a.i.SimpleAsyncUncaughtExceptionHandler : Unexpected exception occurred invoking async method: public void xxxx.handleForPersonSaveEvent(xxxx.PersonSaveEvent) throws javax.security.auth.message.AuthException

08

监听器排序


如果同时有多个监听器监听同一个事件,默认情况下监听器的执行顺序是 随机 的,如果想要他们按照某种顺序执行,可以借助Spring的另外一个注解 @Order 实现。

创建三个监听器,并使用 @Order 排好序。

@Order(1)
@EventListener
public void handleForPersonSaveEvent(PersonSaveEvent saveEvent){
    log.info("handle event1 -> {}", saveEvent);
}

@Order(2)
@EventListener
public void handleForPersonSaveEvent2(PersonSaveEvent saveEvent){
    log.info("handle event2 -> {}", saveEvent);
}

@Order(3)
@EventListener
public void handleForPersonSaveEvent3(PersonSaveEvent saveEvent){
    log.info("handle event3 -> {}", saveEvent);
}

事件发送参照前述示例,发送一个 PersonSaveEvent 事件。

从执行结果可以看到,监听器执行顺序确实是按照 @Order 中指定的顺序执行的。


handle event1 -> PersonSaveEvent(id=1, name=i余数, age=18)
handle event2 -> PersonSaveEvent(id=1, name=i余数, age=18)
handle event3 -> PersonSaveEvent(id=1, name=i余数, age=18)

END


Spring注解 @EventListener 详解

Spring注解 @EventListener 详解

点个在看你最好看

原文始发于微信公众号(i余数):Spring注解 @EventListener 详解

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/194077.html

(0)
小半的头像小半

相关推荐

发表回复

登录后才能评论
极客之音——专业性很强的中文编程技术网站,欢迎收藏到浏览器,订阅我们!