基本概念
01
用途
02
03
处理器
04
返回值
05
异常处理
06
异步监听器
2. 异步监听器的非空返回值不会被当作新的事件发布。如果需要发布新事件,需要注入ApplicationEventPublisher后手动发布。
07
监听器排序
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 "";
}

使用示例
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);
}
}
监听事件
@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);
}
@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);
}
@EventListener(condition = "#root.event.getPayload().getId() == 1")
public void handleByCondition(PersonSaveEvent saveEvent){
log.info("只处理id等于1的 -> {}", saveEvent);
}
只处理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);
}
@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);
}
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
同步监听器异常处理
先定义一个ErrorHandler:
@Slf4j
@Component
public class MyErrorHandler implements ErrorHandler {
@Override
public void handleError(Throwable t) {
log.info("handle error -> {}", t.getClass());
}
}
@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");
}
}
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");
}
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
监听器排序
@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);
}
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

点个
原文始发于微信公众号(i余数):Spring注解 @EventListener 详解
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/194077.html