Spring Event 观察者模型及事件和消息队列之间的区别笔记
Spring Event观察者模型:基于内置事件实现自定义监听
在Spring框架中,观察者模式通过事件驱动模型实现,允许组件间通过事件发布与监听进行解耦通信。这一机制的核心在于
ApplicationEvent、ApplicationListener和ApplicationEventPublisher等接口的协作,结合Spring容器的事件广播器(如ApplicationEventMulticaster),能够高效管理事件的生命周期。
Spring的事件机制通过观察者模式实现了组件间的松耦合通信。开发者可通过自定义事件与监听器灵活扩展业务逻辑,同时利用同步/异步模式优化性能。一般结合事务绑定和泛型支持,可以利用这个机制在复杂系统中展现出强大的灵活性和可维护性。
Spring事件模型的核心组件
事件(ApplicationEvent)
所有事件的基类,自定义事件需继承此类。例如:
public class MyCustomEvent extends ApplicationEvent {private String data;public MyCustomEvent(Object source, String data) {super(source);this.data = data;}}
事件类可携带业务数据,供监听器处理。
事件发布者(ApplicationEventPublisher)
通过
publishEvent()方法发布事件。Spring容器会自动注入ApplicationEventPublisher实例,或通过实现ApplicationEventPublisherAware接口获取:
@Componentpublic class EventPublisher {@Autowiredprivate ApplicationEventPublisher publisher;public void publish(String data) {publisher.publishEvent(new MyCustomEvent(this, data));}}
事件监听器(ApplicationListener)
监听器可通过两种方式实现:
接口实现:继承ApplicationListener并指定泛型事件类型:
@Componentpublic class CustomListener implements ApplicationListener<MyCustomEvent> {@Overridepublic void onApplicationEvent(MyCustomEvent event) {System.out.println("Received: " + event.getData());}}
注解驱动:使用@EventListener标注方法,支持灵活的事件类型匹配:
@Componentpublic class AnnotatedListener {@EventListenerpublic void handleEvent(MyCustomEvent event) {// 处理逻辑}}
事件广播器(ApplicationEventMulticaster)
默认实现类为
SimpleApplicationEventMulticaster,负责将事件分发给所有匹配的监听器。可通过配置其TaskExecutor实现异步事件处理。
实现自定义事件监听的步骤
定义事件类
继承
ApplicationEvent并封装业务数据,例如订单创建事件:
public class OrderCreatedEvent extends ApplicationEvent {private Order order;public OrderCreatedEvent(Object source, Order order) {super(source);this.order = order;}}
发布事件
在业务逻辑中注入
ApplicationEventPublisher并调用其publishEvent()方法:
@Servicepublic class OrderService {@Autowiredprivate ApplicationEventPublisher publisher;public void createOrder(Order order) {// 业务逻辑publisher.publishEvent(new OrderCreatedEvent(this, order));}}
监听事件
方式一:实现接口
适用于需要强类型事件绑定的场景:
@Componentpublic class OrderListener implements ApplicationListener<OrderCreatedEvent> {@Overridepublic void onApplicationEvent(OrderCreatedEvent event) {sendNotification(event.getOrder());}}
方式二:使用注解
支持方法参数自动推导事件类型,更简洁:
@Componentpublic class NotificationService {@EventListenerpublic void onOrderCreated(OrderCreatedEvent event) {// 发送通知}}
异步处理配置
默认事件处理是同步的。若要异步执行,需配置广播器:
@Configurationpublic class AsyncEventConfig {@Beanpublic ApplicationEventMulticaster eventMulticaster() {SimpleApplicationEventMulticaster multicaster = new SimpleApplicationEventMulticaster();multicaster.setTaskExecutor(new SimpleAsyncTaskExecutor());return multicaster;}}
高级特性与注意事项
事务绑定事件(@TransactionalEventListener)
若事件需在事务提交后触发,可使用此注解,并指定
phase参数:
@TransactionalEventListener(phase = TransactionPhase.AFTER_COMMIT)public void handleAfterCommit(OrderPaidEvent event) {// 事务提交后的处理}
TransactionPhase枚举类
package org.springframework.transaction.event;public enum TransactionPhase {BEFORE_COMMIT,// 在提交之前AFTER_COMMIT,// 在提交之后AFTER_ROLLBACK,// 在回滚之后AFTER_COMPLETION;// 在完成之后private TransactionPhase() {}
}
泛型事件处理
Spring通过ResolvableType支持泛型事件的分发,例如监听EntityCreatedEvent<Order>。
异常处理
通过实现
ErrorHandler接口,可统一处理监听器中的异常。
作用
解耦业务逻辑:例如用户注册后发送邮件与短信,通过事件拆分可避免代码耦合。
异步任务处理:如日志记录、数据同步等耗时操作,通过异步监听提升性能。
系统状态监听:例如容器启动(ContextRefreshedEvent)时初始化缓存。
事件(ApplicationEvent)与MQ的核心区别解析
在软件架构中, 事件(ApplicationEvent) 和 消息队列(MQ) 均用于实现组件间的解耦通信,但两者的设计目标、技术实现和适用场景存在显著差异。
事件机制与MQ的本质区别在于通信边界和可靠性设计。事件机制是轻量级的进程内解耦工具,适合单体应用;MQ则是分布式系统中实现高可靠、跨系统通信的基石。
作用范围与通信边界
| 维度 | 事件(ApplicationEvent) | MQ(如RabbitMQ、Kafka) |
|---|---|---|
| 通信范围 | 仅限同一Spring容器内的组件通信(单体应用内部) | 支持跨系统、跨进程、跨语言的分布式通信 |
| 数据一致性 | 依赖本地事务,易与Spring事务管理器集成(如@TransactionalEventListener) | 需依赖MQ的事务消息或最终一致性机制(如RabbitMQ的Confirm模式) |
| 适用场景 | 单体应用内的模块解耦(如用户注册后触发邮件发送) | 分布式系统间的异步通信(如订单系统与库存系统的交互) |
核心差异:
事件机制是进程内通信工具,基于观察者模式实现轻量级解耦;MQ是跨进程/跨系统的中间件,通过消息代理实现分布式系统间的可靠通信。
技术实现与架构特性
通信模型
事件机制:
基于Spring的ApplicationEventPublisher发布事件,ApplicationListener或@EventListener监听处理。
默认同步执行(监听器按顺序触发),但可通过ApplicationEventMulticaster配置异步线程池实现异步处理。
MQ:
采用生产者-消费者模型,依赖Broker(如RabbitMQ的Exchange/Queue)进行消息路由。
天然支持异步通信,消费者可独立处理消息,实现流量削峰和负载均衡。
可靠性
事件机制:
无持久化机制,若系统崩溃或监听器抛出异常,事件可能丢失。
需通过ErrorHandler自定义异常处理逻辑。
MQ:
提供消息持久化、重试机制、死信队列(DLQ)等保障可靠性。
例如RabbitMQ支持消息确认(ACK/NACK)和事务消息,确保消息至少被消费一次。
扩展性
事件机制:
监听器数量增加可能导致性能下降(同步模式下),需谨慎设计异步处理。
无法直接扩展为分布式系统。
MQ:
支持横向扩展(如Kafka的分区机制),可应对高并发场景。
天然适应微服务架构,支持服务间的独立部署与升级。
开发复杂度与生态支持
| 维度 | 事件(ApplicationEvent) | MQ |
|---|---|---|
| 集成成本 | 无需额外依赖,Spring原生支持 | 需引入MQ客户端库、配置连接池等 |
| 学习曲线 | 简单,仅需掌握Spring事件模型(事件类、发布者、监听器) | 需理解MQ协议(如AMQP)、Broker配置、消息模型等 |
| 生态工具 | 局限于Spring生态(如@TransactionalEventListener) | 支持多语言客户端、监控工具(如RabbitMQ Management UI) |
典型用例:
事件机制:订单状态变更时更新缓存。
MQ:电商系统中订单创建后向物流系统推送消息。
性能与资源消耗
吞吐量
事件机制:
同步模式下受限于单线程处理能力,异步模式下性能提升但需管理线程池。
MQ:
支持批量消息处理和消费者集群,吞吐量可达百万级(如Kafka)。
资源占用
事件机制:内存中传递事件对象,无网络开销。
MQ:需维护网络连接、序列化/反序列化数据,资源消耗较高。
适用场景总结
| 场景 | 推荐方案 | 理由 |
|---|---|---|
| 单体应用内的模块解耦 | Spring事件机制 | 轻量级、无外部依赖,适合快速开发 |
| 分布式系统间通信 | MQ | 跨系统支持、高可靠性,适应微服务架构 |
| 事务一致性要求高的操作 | 事件机制+事务监听器 | 通过@TransactionalEventListener实现事务提交后触发 |
| 高并发、异步任务处理 | MQ | 利用消息堆积能力实现削峰填谷 |
| 需要历史消息追溯的场景 | MQ(如Kafka日志留存) | 支持消息持久化和回溯 |
混合使用策略
在实际项目中,二者可互补:
事件机制处理本地逻辑:例如在订单服务内部,使用事件更新缓存或记录日志。
MQ传递跨系统消息:订单创建后,通过MQ通知库存系统和支付系统。
事务协同:通过@TransactionalEventListener确保本地事务提交后,再向MQ发送消息,避免数据不一致。
相关文章:
Spring Event 观察者模型及事件和消息队列之间的区别笔记
Spring Event观察者模型:基于内置事件实现自定义监听 在Spring框架中,观察者模式通过事件驱动模型实现,允许组件间通过事件发布与监听进行解耦通信。这一机制的核心在于ApplicationEvent、ApplicationListener和ApplicationEventPublisher等接…...
【Nova UI】三、探秘 BEM:解锁前端 CSS 命名的高效密码
序言 在上一篇文章中,我们一步一个脚印,扎实地完成了 Vue 组件库搭建的环境搭建工作,从 pnpm 的精妙运用到 TypeScript 的细致配置✍️,每个环节都为组件库的诞生筑牢根基。现在,当我们把目光聚焦到组件库的样式设计时…...
Qt中存储多规则形状图片
在Qt中,您可以通过多种方式处理和存储具有非矩形(多规则形状)的图片。以下是几种主要实现方案: 1. 使用透明通道存储不规则形状 实现方法 // 创建带透明背景的QPixmap QPixmap pixmap(400, 400); pixmap.fill(Qt::transparent);QPainter painter(&…...
前端界面在线excel编辑器 。node编写post接口获取文件流,使用传参替换表格内容展示、前后端一把梭。
首先luckysheet插件是支持在线替换excel内容编辑得但是浏览器无法调用本地文件,如果只是展示,让后端返回文件得二进制文件流就可以了,直接使用luckysheet展示。 这里我们使用xlsx-populate得node简单应用来调用本地文件,自己写一个…...
核心知识——Spark核心数据结构:RDD
引入 通过前面的学习,我们对于Spark已经有一个基本的认识,并且搭建了一个本地的练习环境,因为本专栏的主要对象是数仓和数分,所以就不花大篇幅去写环境搭建等内容,当然,如果感兴趣的小伙伴可以留言&#x…...
Python如何为区块链治理注入智能与高效?
Python如何为区块链治理注入智能与高效? 引言 区块链治理作为一个新兴领域,旨在解决去中心化网络中的决策与协调问题。无论是以太坊的协议升级,还是DAO(去中心化自治组织)内部的投票机制,治理效率与公正性始终是核心挑战。然而,Python的灵活性与强大的生态系统为区块链…...
树莓派 —— 在树莓派4b板卡下编译FFmpeg源码,支持硬件编解码器(mmal或openMax硬编解码加速)
🔔 FFmpeg 相关音视频技术、疑难杂症文章合集(掌握后可自封大侠 ⓿_⓿)(记得收藏,持续更新中…) 正文 1、准备工作 (1)树莓派烧录RaspberryPi系统 (2)树莓派配置固定IP(文末) (3)xshell连接树莓派 (4)...
【Easylive】auditVideo方法详细解析
【Easylive】项目常见问题解答(自用&持续更新中…) 汇总版 auditVideo 方法是视频审核的核心方法,负责处理视频审核状态的变更、用户积分奖励、数据同步以及文件清理等操作。下面我将从功能、流程、设计思路等方面进行全面解析。 1. 方…...
【数据分享】中国3254座水库集水区特征数据集(免费获取)
水库在水循环、碳通量、能量平衡中扮演关键角色,实实在在地影响着我们的生活。其功能和环境影响高度依赖于地理位置、上游流域属性(如地形、气候、土地类型)和水库自身的动态特征(如水位、蒸发量)。但在此之前一直缺乏…...
Maven安装与配置完整指南
Maven安装与配置完整指南 1. 前言 Apache Maven 是一个强大的项目管理和构建工具,广泛应用于Java项目开发。它通过 POM(Project Object Model) 文件管理项目依赖,并提供了标准化的构建流程。 本文详细介绍 Maven的下载、安装、环境配置、镜像加速、IDE集成 以及 常见问题…...
我用Axure画了一个富文本编辑器,还带交互
最近尝试用Axure RP复刻了一个富文本编辑器,不仅完整还原了工具栏的各类功能,还通过交互设计实现了接近真实编辑器操作体验。整个设计过程聚焦功能还原与交互流畅性,最终成果令人惊喜。 编辑器采用经典的三区布局:顶部工具栏集成了…...
Uniapp自定义TabBar组件全封装实践与疑难问题解决方案
前言 在当前公司小程序项目中,我们遇到了一个具有挑战性的需求:根据不同用户身份动态展示差异化的底部导航栏(TabBar) 。这种多角色场景下的UI适配需求,在提升用户体验和实现精细化运营方面具有重要意义。 在技术调研…...
【PCB工艺】软件是如何控制硬件的发展过程
软件与硬件的关系密不可分,软件的需求不断推动硬件的发展,而硬件的进步又为软件创新提供了基础。 时光回溯到1854年,亨利戈培尔发明了电灯泡(1879年,托马斯阿尔瓦爱迪生找到了更合适的材料研制出白炽灯。)…...
Javascript代码压缩混淆工具terser详解
原始的JavaScript代码在正式的服务器上,如果没有进行压缩,混淆,不仅加载速度比较慢,而且还存在安全和性能问题. 因此现在需要进行压缩,混淆处理. 处理方案简单描述一下: 1. 使用 terser 工具进行 安装 terser工具: # npm 安装 npm install terser --save-dev# 或使用 yarn 安…...
【教程】如何利用bbbrisk一步一步实现评分卡
利用bbbrisk一步一步实现评分卡 一、什么是评分卡1.1.什么是评分卡1.2.评分卡有哪些 二、评分卡怎么弄出来的2.1.如何制作评分卡2.2.制作评分卡的流程 三、变量的分箱3.1.数据介绍3.2.变量自动分箱3.3.变量的筛选 四、构建评分卡4.1.评分卡实现代码4.2.评分卡表4.3.阈值表与分数…...
RAG优化:python从零实现Proposition Chunking[命题分块]让 RAG不再“断章取义”,从此“言之有物”!
🧠 向所有学习者致敬! “学习不是装满一桶水,而是点燃一把火。” —— 叶芝 我的博客主页: https://lizheng.blog.csdn.net 🌐 欢迎点击加入AI人工智能社区! 🚀 让我们一起努力,共创AI未来! 🚀 大家好,本篇要聊的是一个让 RAG不再“断章取义”的神奇技术——命…...
丝杆,同步带,链条选型(我要自学网)
这里的选型可以70%的正确率,正确率不高,但是选型速度会比较快。 1.丝杆选型 后面还有一堆计算公式,最终得出的结果是导程25,轴径25mm的丝杆。 丝杆选择长度时,还要注意细长比,长度/直径 一般为30到50。 2…...
【YOLO系列】基于YOLOv8的无人机野生动物检测
基于YOLOv8的无人机野生动物检测 1.前言 在野生动物保护、生态研究和环境监测领域,及时、准确地检测和识别野生动物对于保护生物多样性、预防人类与野生动物的冲突以及制定科学的保护策略至关重要。传统的野生动物监测方法通常依赖于地面巡逻、固定摄像头或无线传…...
一文详细讲解Python(详细版一篇学会Python基础和网络安全)
引言 在当今数字化时代,Python 作为一种简洁高效且功能强大的编程语言,广泛应用于各个领域,从数据科学、人工智能到网络安全等,都能看到 Python 的身影。而网络安全作为保障信息系统和数据安全的关键领域,其重要性不言…...
NFS 重传次数速率监控
这张图展示的是 NFS 重传次数速率监控,具体解释如下: 1. 指标含义 监控指标 node_nfs_rpc_retransmissions_total 统计 NFS(网络文件系统)通信中 RPC(远程过程调用)的重传次数,rate(node_nfs_…...
【Java】Hibernate的一级缓存
Session是有一个缓存, 又叫Hibernate的一级缓存 session缓存是由一系列的Java集合构成的。当一个对象被加入到Session缓存中,这个对象的引用就加入到了java的集合中,以后即使应用程序中的引用变量不再引用该对象,只要Session缓存不被清空&…...
学习笔记--(6)
import numpy as np import matplotlib.pyplot as plt from scipy.special import erfc# 设置参数 rho 0.7798 z0 4.25 # 确保使用大写 Z0,与定义一致def calculate_tau(z, z_prime, rho, s_values):return np.log(rho * z * z_prime * s_values / 2)# 定义 chi_…...
【QT5 网络编程示例】TCP 通信
文章目录 TCP 通信 TCP 通信 QT主要通过QTcpSocket 和 QTcpServer两个类实现服务器和客户端的TCP 通信。 QTcpSocket 是 Qt 提供的套接字类,看用于建立、管理和操作 TCP 连接。 常用方法 connectToHost(host, port):连接到指定服务器。disconnectFro…...
JWT在线解密/JWT在线解码 - 加菲工具
JWT在线解密/JWT在线解码 首先进入加菲工具 选择 “JWT 在线解密/解码” https://www.orcc.top 或者直接进入JWT 在线解密/解码 https://www.orcc.top/tools/jwt 进入功能页面 使用 输入对应的jwt内容,点击解码按钮即可...
【Linux】用户向硬件寄存器写入值过程理解
思考一下,当我们咋用户态向寄存器写入一个值,这个过程是怎么样的呢?以下是应用程序通过标准库函数(如 write()、ioctl() 或 mmap())向硬件寄存器写入值的详细过程,从用户空间到内核再到硬件的完整流程&…...
【Easylive】convertVideo2Ts 和 union 方法解析
【Easylive】项目常见问题解答(自用&持续更新中…) 汇总版 这两个方法是 transferVideoFile 中用于视频文件处理的核心辅助方法,下面我将结合它们在 transferVideoFile 中的使用场景进行详细解释。 1. convertVideo2Ts 方法解析 方法签…...
飞桨PP系列新成员PP-DocLayout开源,版面检测加速大模型数据构建,超百页文档图像一秒搞定
背景介绍 文档版面区域检测技术通过精准识别并定位文档中的标题、文本块、表格等元素及其空间布局关系,为后续文本分析构建结构化上下文,是文档图像智能处理流程的核心前置环节。随着大语言模型、文档多模态及RAG(检索增强生成)等…...
Java 锁机制详解:用“厕所门”和“防盗门”轻松理解多线程同步
Java 锁机制详解:用“厕所门”和“防盗门”轻松理解多线程同步 目录 锁的作用synchronized 关键字ReentrantLockReadWriteLockStampedLock避免死锁的诀窍总结与对比 锁的作用 生活中的例子:公共厕所一次只能进一人,门上的“有人/无人”标志…...
关于修改 vue Element admin、若依, 等后台管理系统模板的一些全局样式问题:
关于修改 vue Element admin、若依, 等后台管理系统模板的一些全局样式问题: 1、修改左侧菜单和顶部(菜单)的背景色、把背景色改为炫酷的背景图。 1)上传图片 src/assets/images/menu-icon.png、 src/assets/images/…...
并发多线程八股
并发多线程 1.Java里面的线程和操作系统的线程一样吗?2.Java的线程安全在三个方面体现:3.保证数据一致性的方案4.线程创建的方式1)Thread类2)Runnable接口3)Callable接口和FutureTask4)线程池(e…...
