Spring事件注解@EventListener【观察】
一、背景
在开发工作中,我们常常会遇到这样一种情况:完成一项任务后,需要向其他模块广播消息或通知,以触发其他事件的处理。逐个发送请求固然可行,但更好的方式是采用事件监听,它是设计模式中的发布-订阅模式和观察者模式的一种实现。
观察者模式简单来说就是一种角色扮演,你在做某件事时有人在一旁观察你。当这个人观察到你有兴趣的特定事件发生时,他们会根据这个事件做一些其他的事情。但请注意,任何想要观察你的人都必须先到你这儿进行登记,否则你将无法通知到他们,或者他们没有资格来观察你。
在Spring容器中,我们可以监听并触发各种事件。通常有两种方法可以实现这一目标:使用ApplicationListener接口和使用@EventListener注解。这两种方法都能帮助我们更好地管理和响应应用程序中的各种事件。
二、简介
2.1 用途
为了实现一个能够监听应用程序事件的监听器方法,我们可以使用特定的注解来标记该方法,并定义其支持的事件类型。
如果监听器方法只支持单一的事件类型,我们可以将其参数声明为该事件类型的唯一实例。例如,如果我们的监听器方法只监听ApplicationEvent实例,则可以将其参数声明为ApplicationEvent类型。
如果监听器方法支持多种事件类型,我们可以使用注解的classes属性来指定一个或多个支持的事件类型。这样,该方法就可以监听到在classes属性中指定的任何事件类型。
2.2 事件处理条件
可以通过 condition 属性指定一个SpEL表达式,如果返回 “true”, “on”, “yes”, or “1” 中的任意一个,则事件会被处理,否则不会。
2.3 处理器
在Spring框架中,@EventListener注解的处理是通过内部的EventListenerMethodProcessor类进行的。这个类负责寻找带有@EventListener注解的方法,并在适当的时候触发它们。
当使用Java配置时,EventListenerMethodProcessor会自动注册到Spring容器中。你只需要在带有@EventListener注解的方法上添加@Component或@Service等注解,Spring就会自动检测并注册这个方法。
2.4 返回值
被标注的方法可以没有返回值,也可以有返回值。当有返回值是,其返回值会被当作为一个新的事件发送。如果返回类型是数组或集合,那么数组或集合中的每个元素都作为一个新的单独事件被发送。
2.5 异常处理
同步监听器抛出的所有checked异常都会被封装成 UndeclaredThrowableException ,因为事件发布者只能处理运行时异常(unchecked异常)。
2.6 异步监听器
当需要异步处理监听器时,可以在监听器方法上再增加另外的一个Spring注解 @Async,但需要注意以下限制:
监听器报错不会传递给事件发起者,因为双方已经不在同一个线程了。
异步监听器的非空返回值不会被当作新的事件发布。如果需要发布新事件,需要注入 ApplicationEventPublisher后手动发布。
2.7 监听器排序
如果同一个事件可能会被多个监听器监听处理,那么我们可以使用 @Order 注解对各个监听器进行排序。
2.8 源码
@Target({ElementType.METHOD, ElementType.ANNOTATION_TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface EventListener {/*** Alias for {@link #classes}.*/@AliasFor("classes")Class<?>[] value() default {};/*** 可以处理的事件类型*/@AliasFor("value")Class<?>[] classes() default {};/*** SpEL表达式判断是否满足处理条件*/String condition() default "";/*** 可以给监听器指定一个id,默认是方法的全限定名,如:mypackage.MyClass.myMethod()*/String id() default "";}
三、示例详解
3.1 单一事件监听器
发布事件
@Service
public class EventPublisher {private ApplicationEventPublisher eventPublisher;@Autowiredpublic 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 {@EventListenerpublic void handleForPersonSaveEvent(PersonSaveEvent saveEvent){log.info("saveEvent -> {}", saveEvent);}
}
结果验证
saveEvent -> PersonSaveEvent(id=1, name=i余数, age=18)
3.2 使用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)
3.3 使用condition筛选监听的事件
发布事件
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为2的事件不满足条件,所以不会执行。
只处理id等于1的 -> PersonSaveEvent(id=1, name=i余数, age=18)
3.4 有返回值的监听器
3.4.1 返回一个单一对象
发布事件
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)
3.4.2 返回一个集合
将监听器稍作修改,使其返回一个集合。
@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)
3.4.3 返回一个数组
和返回值为集合一样,数组中的每个对象都被当作一个单独的事件进行发送。
3.4.4 异步监听器
创建两个监听器,一个同步一个异步,异步监听器就是在方法上加一个 @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)
3.5 监听器异常处理
3.5.1 同步异常处理
使用 SimpleApplicationEventMulticaster 处理同步监听器抛出异常。
先定义一个ErrorHandler:
@Slf4j
@Component
public class MyErrorHandler implements ErrorHandler {@Overridepublic void handleError(Throwable t) {log.info("handle error -> {}", t.getClass());}
}
将自定义ErrorHandler绑定到 SimpleApplicationEventMulticaster。
@Slf4j
@Service
public class EventListenerService {@Autowiredprivate SimpleApplicationEventMulticaster simpleApplicationEventMulticaster;@Autowiredprivate MyErrorHandler errorHandler;@PostConstructpublic void init(){simpleApplicationEventMulticaster.setErrorHandler(errorHandler);}@Order(1)@EventListenerpublic 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
3.5.2 异步异常处理
使用 SimpleAsyncUncaughtExceptionHandler 来处理 @Async 抛出的异常。
@Configuration
public class AsyncConfig implements AsyncConfigurer {@Overridepublic 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
3.6 监听器排序
如果同时有多个监听器监听同一个事件,默认情况下监听器的执行顺序是随机的,如果想要他们按照某种顺序执行,可以借助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);
}
从执行结果可以看到,确实是按照@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)
四、总结
在Spring框架中,@EventListener注解用于标识监听器方法,可以监听Spring ApplicationEvent或其他任意对象的事件。
当一个方法被标注为@EventListener时,它就被视为一个事件监听器,可以接收和响应特定类型的事件。这个方法可以在事件发布时被调用,以处理该事件。在Spring框架中,EventListenerMethodProcessor负责寻找和调用带有@EventListener注解的方法。
@EventListener注解可以设置两个属性:value和classes,用于指定监听器监听的事件类型。如果value和classes没有设置值,那么被标注的方法必须有一个且只能有一个参数。如果value或classes设置了值,单个Class值表示监听器可以不用设置参数;多个Class值则要求被标注的方法一定不要设置参数。这是因为在监听事件时,通过反射监听方法处理事件,事件转换成方法参数时,若类型不一致,可能会发生类型转换异常。
此外,还有一个注解@TransactionalEventListener,它继承了@EventListener的功能,并添加了新的特性。这个注解可以用于在事务处理期间监听事件,并且可以根据需要自动回滚事务。
总的来说,EventListener是一种非常有用的技术,可以帮助我们更好地管理和响应应用程序中的事件。在Spring框架中,我们可以使用@EventListener注解来方便地创建和注册事件监听器,从而实现更高效的事件处理和控制。
相关文章:
Spring事件注解@EventListener【观察】
一、背景 在开发工作中,我们常常会遇到这样一种情况:完成一项任务后,需要向其他模块广播消息或通知,以触发其他事件的处理。逐个发送请求固然可行,但更好的方式是采用事件监听,它是设计模式中的发布-订阅模…...
玩转Spring中强大的spel表达式!
玩转Spring中强大的spel表达式!值得推荐的好文:https://zhuanlan.zhihu.com/p/174786047...
HTTPS攻击原理 被攻击该如何防护
简单来说,HTTPS HTTP SSL/TLS。 在 HTTP 协议中,客户端通过网络传输消息与服务器进行通信。但该消息采用明文的原始格式。坏人(攻击者)很容易窃听消息。这就是我们需要 SSL/TLS 的原因。 HTTPS是一种安全的HTTP协议,…...
【.NET Core】委托(Delegate)应用详解
【.NET Core】委托(Delegate)应用详解 文章目录 【.NET Core】委托(Delegate)应用详解一、概述二、委托(Delegate)定义三、基础委托(Delegate) - 无返回值委托四、基础委托(Delegate) - 有返回值委托五、Mu…...

【LeetCode:1457. 二叉树中的伪回文路径 | 二叉树 + DFS +回文数】
🚀 算法题 🚀 🌲 算法刷题专栏 | 面试必备算法 | 面试高频算法 🍀 🌲 越难的东西,越要努力坚持,因为它具有很高的价值,算法就是这样✨ 🌲 作者简介:硕风和炜,…...

《golang设计模式》第三部分·行为型模式-06-备忘录模式(Memento)
文章目录 1. 概述1.1 角色1.2 类图 2. 代码示例2.1 设计2.2 代码2.3 类图 1. 概述 备忘录(Memento)用于在不破坏目标对象封装特性的基础上,将目标对象内部的状态存储到外部对象中,以备之后恢复状态时使用。 1.1 角色 Originato…...

Cache学习(4):Cache分配策略Cache更新策略Cache逐出策略
Cache的数据流 常用名词 Allocation 分配Eviction 驱逐分配策略和更新策略分别为当产生Cache miss和Cache hit的时候数据流的具体行为 1 Cache分配策略(Cache Allocation Policy) Cache的分配策略是指不同情况下为数据分配Cache Line的不同行为。Cac…...
角色管理--产品经理岗
研发组织管理--角色管理--产品经理岗 定位 相对稳定和简单产品的独立产品打造者,复杂产品的辅助者 所需资质 校招新人,拥有灵性拥有基础的产品力(认知,设计,创新,推进,学习)Axur…...
SQL数据迁移实战:从产品层级信息到AB测试表
文章目录 创建表插入数据清空数据表数据迁移和筛选查询数据结论 创建表 首先,代码中定义了两个表格:dim_prod_hierarchy_info 和 app_abtest_product_info,都位于 test 数据库中。 dim_prod_hierarchy_info 表用于存储产品层级信息…...
VMware系列:VMware安装Android虚拟机
VMware系列:VMware安装Android虚拟机 一. 下载镜像这里提供了三种下载镜像方式,也就是三个下载链接,这里推荐百度网盘下载二. 使用VMware Workstation Pro 创建新的虚拟机操作系统应该可以选择任意一个,笔者只试过下图中,如果读者感兴趣可以多试几个,但笔者不保证每个都可…...

链接1:编译器驱动程序
文章目录 GNU编译器示例编译 GNU编译器 GNU编译器(GNU Compiler)是由自由软件基金会(Free Software Foundation,FSF)开发和维护的一套编译器集合。这些编译器主要用于编译各种编程语言的源代码,将其转换为…...

经典滑动窗口试题(二)
📘北尘_:个人主页 🌎个人专栏:《Linux操作系统》《经典算法试题 》《C》 《数据结构与算法》 ☀️走在路上,不忘来时的初心 文章目录 一、水果成篮1、题目讲解2、讲解算法思路3、代码实现 二、找到字符串中所有字母异位词1、题目…...

easyexcel指定sheet页动态给行列加背景色
需求 1、easyexcel,有多个sheet页,某些sheet页的行、列动态需要加背景色。 2、扩展支持cellStyle标记单元格超过64000 import com.alibaba.excel.metadata.CellData; import com.alibaba.excel.metadata.Head; import com.alibaba.excel.write.handler.…...
设计模式在实际业务中应用 - 模版方法
1. 业务背景 作者在工作中主要主导 A 业务线的系统建设,A 业务线主要是零售场景酒水的售卖与即时配送服务。为了方便运营在自研系统中对多平台商品进行管理而开发的三方平台商品管理功能,本次介绍的模版方法模式则是在该功能开发过程中的落地实践。 2.…...

BGP综合实验
任务如下: 1.AS1存在两个环回,一个地址为192.168.1.0/24该地址不能在任何协议中宣告 AS3存在两个环回,一个地址为192.168.2.0/24该地址不能在任何协议中宣告,最终要求这两个环回可以互相通讯 2.整个AS2的IP地址为172.16.0.0/16&…...

Global Surface Summary of the Day 全球逐日气象站点数据 GSOD数据集
数据名称 Global Surface Summary of the Day 数据内容 数据包含以下气象要素的日值观测数据: 气压:平均气压、海平面气压;气温:平均气温、日最高气温、日最低气温;湿度:露点温度(需自行换算…...

Harmony OS4开发入门
代码地址: https://gitee.com/BruceLeeAdmin/harmonyos/tree/master 项目目录介绍 ArkTS介绍 简单案例: State times: number 0/*数据类型:stringnumberany: 不确定类型,可以是任意类型*/State msg: string "hello"…...
.net core 事务
在 .NET Core 中,可以使用 Entity Framework Core 来实现事务处理。下面是一个简单的示例,展示了如何在 .NET Core 中使用 Entity Framework Core 来创建和执行事务: using System; using Microsoft.EntityFrameworkCore; using System.Tran…...

【Python】python天气数据抓取与数据分析(源码+论文)【独一无二】
👉博__主👈:米码收割机 👉技__能👈:C/Python语言 👉公众号👈:测试开发自动化【获取源码商业合作】 👉荣__誉👈:阿里云博客专家博主、5…...

MPPT工作流程及算法和硬件的选择
MPPT算法选择 目前,MPPT算法有开路电压比率(离线)、短路电流比率(离线)、观察调节(在线)、极限追踪控制法(在线)。 在光伏控制系统中,因为日照、温度等条件的变化,光伏电池的输出功率也是在不断变化的,为保证使得光伏电池的输出功…...

基于距离变化能量开销动态调整的WSN低功耗拓扑控制开销算法matlab仿真
目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.算法仿真参数 5.算法理论概述 6.参考文献 7.完整程序 1.程序功能描述 通过动态调整节点通信的能量开销,平衡网络负载,延长WSN生命周期。具体通过建立基于距离的能量消耗模型&am…...
【Linux】C语言执行shell指令
在C语言中执行Shell指令 在C语言中,有几种方法可以执行Shell指令: 1. 使用system()函数 这是最简单的方法,包含在stdlib.h头文件中: #include <stdlib.h>int main() {system("ls -l"); // 执行ls -l命令retu…...

高频面试之3Zookeeper
高频面试之3Zookeeper 文章目录 高频面试之3Zookeeper3.1 常用命令3.2 选举机制3.3 Zookeeper符合法则中哪两个?3.4 Zookeeper脑裂3.5 Zookeeper用来干嘛了 3.1 常用命令 ls、get、create、delete、deleteall3.2 选举机制 半数机制(过半机制࿰…...

江苏艾立泰跨国资源接力:废料变黄金的绿色供应链革命
在华东塑料包装行业面临限塑令深度调整的背景下,江苏艾立泰以一场跨国资源接力的创新实践,重新定义了绿色供应链的边界。 跨国回收网络:废料变黄金的全球棋局 艾立泰在欧洲、东南亚建立再生塑料回收点,将海外废弃包装箱通过标准…...

ardupilot 开发环境eclipse 中import 缺少C++
目录 文章目录 目录摘要1.修复过程摘要 本节主要解决ardupilot 开发环境eclipse 中import 缺少C++,无法导入ardupilot代码,会引起查看不方便的问题。如下图所示 1.修复过程 0.安装ubuntu 软件中自带的eclipse 1.打开eclipse—Help—install new software 2.在 Work with中…...

(转)什么是DockerCompose?它有什么作用?
一、什么是DockerCompose? DockerCompose可以基于Compose文件帮我们快速的部署分布式应用,而无需手动一个个创建和运行容器。 Compose文件是一个文本文件,通过指令定义集群中的每个容器如何运行。 DockerCompose就是把DockerFile转换成指令去运行。 …...
Rapidio门铃消息FIFO溢出机制
关于RapidIO门铃消息FIFO的溢出机制及其与中断抖动的关系,以下是深入解析: 门铃FIFO溢出的本质 在RapidIO系统中,门铃消息FIFO是硬件控制器内部的缓冲区,用于临时存储接收到的门铃消息(Doorbell Message)。…...

GC1808高性能24位立体声音频ADC芯片解析
1. 芯片概述 GC1808是一款24位立体声音频模数转换器(ADC),支持8kHz~96kHz采样率,集成Δ-Σ调制器、数字抗混叠滤波器和高通滤波器,适用于高保真音频采集场景。 2. 核心特性 高精度:24位分辨率,…...

Linux --进程控制
本文从以下五个方面来初步认识进程控制: 目录 进程创建 进程终止 进程等待 进程替换 模拟实现一个微型shell 进程创建 在Linux系统中我们可以在一个进程使用系统调用fork()来创建子进程,创建出来的进程就是子进程,原来的进程为父进程。…...
大语言模型(LLM)中的KV缓存压缩与动态稀疏注意力机制设计
随着大语言模型(LLM)参数规模的增长,推理阶段的内存占用和计算复杂度成为核心挑战。传统注意力机制的计算复杂度随序列长度呈二次方增长,而KV缓存的内存消耗可能高达数十GB(例如Llama2-7B处理100K token时需50GB内存&a…...