当前位置: 首页 > news >正文

Spring Boot 事件驱动:构建灵活可扩展的应用

  在 Spring Boot 应用中,事件发布和监听机制是一种强大的工具,它允许不同的组件之间以松耦合的方式进行通信。这种机制不仅可以提高代码的可维护性和可扩展性,还能帮助我们构建更加灵活、响应式的应用。本文将深入探讨 Spring Boot 的事件发布和监听机制,揭示其工作原理,并分享一些最佳实践。

一、观察者模式的应用

  Spring Boot的事件发布与监听机制基于观察者模式(Observer Pattern),这是一种行为设计模式,其中对象之间定义了一对多的依赖关系,当一个对象的状态发生变化时,所有依赖于它的对象都会自动得到通知并更新。具体来说,在Spring Boot中,ApplicationEvent作为事件对象,而ApplicationListener则充当事件监听器的角色。每当某个事件发生时,Spring容器会负责通知所有的监听器,使得它们能够根据接收到的信息执行相应的操作。

二、核心概念

  1. 事件(Event):事件是应用中发生的某个状态变化的信号。在 Spring Boot 中,事件通常是一个继承自 ApplicationEvent 的 Java 对象。事件可以是系统事件(例如,应用启动事件、关闭事件),也可以是自定义事件(例如,用户注册事件、订单创建事件)。
  2. 事件发布者(Event Publisher):事件发布者负责创建并发布事件。在 Spring Boot 中,通常通过 ApplicationEventPublisher 接口来实现事件的发布。
  3. 事件监听器(Event Listener):事件监听器负责监听特定的事件,并在事件发生时执行相应的处理逻辑。在 Spring Boot 中,通常通过 @EventListener 注解或者实现 ApplicationListener 接口来实现事件的监听。

三、实现方式

3.1 定义事件

import org.springframework.context.ApplicationEvent;public class MyCustomEvent extends ApplicationEvent {private String message;public MyCustomEvent(Object source, String message) {super(source);this.message = message;}public String getMessage() {return message;}
}

3.2 发布事件

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationEventPublisher;
import org.springframework.stereotype.Service;@Service
public class MyService {@Autowiredprivate ApplicationEventPublisher publisher;public void doSomething() {//发布事件publisher.publishEvent(new MyCustomEvent(this, "触发自定义事件!"));}
}

3.3 监听事件

  这可以通过两种方式实现:

  • 实现ApplicationListener接口:这种方式较为传统,需要显式地实现onApplicationEvent()方法。
import org.springframework.context.ApplicationListener;
import org.springframework.stereotype.Component;@Component
public class MyEventListener implements ApplicationListener<MyCustomEvent> {@Overridepublic void onApplicationEvent(MyCustomEvent event) {System.out.println("接收到自定义事件: " + event.getMessage());}
}

  值得注意的是,对于同步监听器而言,如果希望控制不同监听器之间的执行顺序,则可以使用@Order注解指定优先级。在不同的监听器类上面加上@Order(1)、@Order(2)、@Order(3)…即可

接收到自定义事件1: 触发自定义事件!
接收到自定义事件2: 触发自定义事件!
接收到自定义事件3: 触发自定义事件!
  • 使用@EventListener注解:这是更为简洁的做法,只需在目标方法上添加此注解即可。此外,还可以结合@Async注解实现异步监听,但需确保已启用异步支持(例如,在主类或配置类上添加@EnableAsync)。示例代码如下所示:
import org.springframework.context.event.EventListener;
import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Component;@Component
public class MyEventListener {@EventListener@Async //使用@Async注解,进行异步处理public void handleMyCustomEvent(MyCustomEvent event) {System.out.println("接收到自定义事件: " + event.getMessage());}
}

四、工作原理

  • 事件发布:当调用 ApplicationEventPublisher.publishEvent() 方法发布事件时,Spring Boot 会根据事件的类型,将事件发送给所有注册的监听器。如果配置了异步监听,则会在另外的线程处理。

  • 事件监听:Spring Boot 会在应用启动时,扫描所有带有 @EventListener 注解的方法或者实现了 ApplicationListener 接口的类,并将它们注册为事件监听器。当事件发生时,Spring Boot 会根据事件类型,调用相应的监听器的方法。监听器方法可以通过参数获取事件对象,并执行相应的处理逻辑。

五、应用场景

  1. 用户注册:用户注册成功后,发布用户注册事件,触发发送欢迎邮件、短信通知等操作。
  2. 订单创建:订单创建完成后,发布订单创建事件,触发扣减库存、发送订单通知等操作。
  3. 日志记录:应用发生异常时,发布异常事件,记录异常信息到日志文件或数据库中。
  4. 系统监控:应用启动或关闭时,发布启动或关闭事件,用于系统监控。
  5. 模块解耦:不同模块之间通过事件进行通信,实现模块之间的解耦。

六、高级特性

  除了基本的功能外,Spring Boot还提供了其他一些高级特性来丰富事件发布与监听机制的功能性:

  • 条件监听:借助SpEL表达式,我们可以为@EventListener设置条件属性,只有当满足指定条件时才会触发监听器。例如,仅当用户的名字为“张三”时才处理用户更新事件。
  • 事务感知:通过@TransactionalEventListener注解,可以使监听器的行为与事务状态相关联,默认情况下是在事务提交后执行监听器逻辑。
  • 多播器(Multicaster):在内部,Spring使用ApplicationEventMulticaster来管理和分发事件给多个监听器,确保每个监听器都能接收到正确的事件。

七、最佳实践

  1. 合理定义事件: 事件应该足够细粒度,能够准确描述应用中发生的状态变化。
  2. 避免过度使用事件:事件机制主要用于处理异步操作,对于同步操作,应该尽量使用直接方法调用。
  3. 使用事件发布者接口: 为了代码的简洁和清晰,尽量使用ApplicationEventPublisher 接口发布事件,避免手动创建事件对象。
  4. 避免在事件监听器中执行耗时操作:如果需要在事件监听器中执行耗时操作,应该将其放入单独的线程池中执行,或者使用 @Async 注解进行异步处理。
  5. 注意事件的传递:如果事件需要在不同的组件之间传递,应该考虑事件的传递效率和安全性。
  6. 测试: 编写单元测试或集成测试,确保事件发布和监听机制的正确性。
  7. 异常处理: 在事件监听器中,做好异常处理,避免影响其他业务功能,可以使用try catch 捕获异常,也可以使用@ExceptionHandler 来统一处理异常。

相关文章:

Spring Boot 事件驱动:构建灵活可扩展的应用

在 Spring Boot 应用中&#xff0c;事件发布和监听机制是一种强大的工具&#xff0c;它允许不同的组件之间以松耦合的方式进行通信。这种机制不仅可以提高代码的可维护性和可扩展性&#xff0c;还能帮助我们构建更加灵活、响应式的应用。本文将深入探讨 Spring Boot 的事件发布…...

IM系统设计

读多写少&#xff0c;一般采用写扩散成timeline来做 写扩散模式 利用last message id作为这个作为最后一个消息体 timeline和批量未读和ack 利用ZSET来维护连接的定时心跳&#xff0c;来续约运营商的连接不断开...

华为EC6110T-海思Hi3798MV310_安卓9.0_通刷-强刷固件包

华为EC6110T-海思Hi3798MV310_安卓9.0_通刷-强刷固件包 刷机教程说明&#xff1a; 适用机型&#xff1a;华为EC6110-T、华为EC6110-U、华为EC6110-M 破解总分为两个部分&#xff1a;拆机短接破解&#xff08;保留IPTV&#xff09;和OTT卡刷&#xff08;不保留IPTV&#xff09…...

ASP.NET Blazor托管模型有哪些?

今天我们来说说Blazor的三种部署方式&#xff0c;如果大家还不了解Blazor&#xff0c;那么我先简单介绍下Blazor Blazor 是一种 .NET 前端 Web 框架&#xff0c;在单个编程模型中同时支持服务器端呈现和客户端交互性&#xff1a; ● 使用 C# 创建丰富的交互式 UI。 ● 共享使用…...

PyTorch广告点击率预测(CTR)利用深度学习提升广告效果

目录 广告点击率预测问题数据集结构广告点击率预测模型的构建1. 数据集准备2. 构建数据加载器3. 构建深度学习模型4. 训练与评估 总结 广告点击率预测&#xff08;CTR&#xff0c;Click-Through Rate Prediction&#xff09;是在线广告领域中的重要任务&#xff0c;它帮助广告平…...

PAT甲级-1017 Queueing at Bank

题目 题目大意 银行有k个窗口&#xff0c;每个窗口只能服务1个人。如果3个窗口已满&#xff0c;就需要等待。给出n个人到达银行的时间和服务时间&#xff0c;要求计算每个人的平均等待时间。如果某个人的到达时间超过17:00:00&#xff0c;则不被服务&#xff0c;等待时间也不计…...

OneData体系架构详解

阿里巴巴的 OneData 体系架构方法论&#xff0c;主要分为三个阶段&#xff1a;业务板块、规范定义 和 模型设计。每个阶段的核心目标是确保数据的高效管理、共享与分析能力。 一. 业务板块&#xff08;Business Segment&#xff09; 业务板块是OneData体系架构中的第一步&…...

Gin 框架入门实战系列教程

一&#xff0c;Gin介绍 Gin是一个 Go (Golang) 编写的轻量级 http web 框架&#xff0c;运行速度非常快&#xff0c;如果你是性能和高效的追求者&#xff0c;我们推荐你使用Gin框架。 Gin最擅长的就是Api接口的高并发&#xff0c;如果项目的规模不大&#xff0c;业务相对简单…...

鸿蒙harmony json转对象(2)

在ArkTS&#xff08;Ark TypeScript&#xff09;中&#xff0c;接口&#xff08;interface&#xff09;是用来定义一个对象的结构&#xff0c;它可以包含属性、方法签名&#xff0c;以及嵌套的类型&#xff08;包括其他接口或对象类型&#xff09;。因此&#xff0c;接口里面可…...

M-LAG与E-trunk

M-LAG和E-trunk都是用来实现跨设备链路聚合&#xff0c;解决单点故障的&#xff0c;其大部分特性相同&#xff0c;工作模式M-LAG更胜一筹&#xff0c;支持双活,而且其原理感觉像是vrrpmstp的升级版&#xff0c;是往增加网络可靠性去发展的;而E-trunk是基于LACP扩展实现&#xf…...

【面试常见问题】

如何自我介绍 自我介绍是面试关键部分&#xff0c;是面试官了解求职者的首要途径&#xff0c;清晰自信的介绍能提升面试官印象&#xff0c;对求职成功至关重要。 糟糕的自我介绍示例 求职者朱晓明虽表明自己善于交际、积极&#xff0c;23 年毕业且从事 java 开发&#xff0c…...

Spring Boot Starter介绍

前言 大概10来年以前&#xff0c;当时springboot刚刚出现并没有流行&#xff0c;当时的Java开发者们开发Web应用主要是使用spring整合springmvc或者struts、iBatis、hibernate等开发框架来进行开发。项目里一般有许多xml文件配置&#xff0c;其中配置了很多项目中需要用到的Be…...

vue和reacts数据响应式的差异

Vue 的数据响应式&#xff1a; 原理&#xff1a; Vue 使用 Object.defineProperty 或 Proxy&#xff08;在 Vue 3 中&#xff09;来实现数据的响应式。当创建 Vue 实例时&#xff0c;会对 data 对象中的属性进行遍历&#xff0c;将其转换为响应式属性。对于 Object.definePro…...

OpenEuler学习笔记(九):安装 OpenEuler后配置和优化

安装OpenEuler后&#xff0c;可以从系统基础设置、网络配置、性能优化等方面进行配置和优化&#xff0c;以下是具体内容&#xff1a; 系统基础设置 更新系统&#xff1a;以root用户登录系统后&#xff0c;在终端中执行sudo yum update命令&#xff0c;对系统进行更新&#xf…...

npm命令与yarn命令的区别

npm与Yarn的区别详解 在软件开发中&#xff0c;npm和Yarn都是流行的包管理工具&#xff0c;它们各自拥有独特的特性和优势。以下是它们的主要区别&#xff1a; 1. 安装速度 npm&#xff1a;安装速度相对较慢&#xff0c;尤其是在依赖项较多的情况下。Yarn&#xff1a;采用并…...

python如何导出数据到excel文件

python导出数据到excel文件的方法&#xff1a; 1、调用Workbook()对象中的add_sheet()方法 wb xlwt.Workbook() ws wb.add_sheet(A Test Sheet) 2、通过add_sheet()方法中的write()函数将数据写入到excel中&#xff0c;然后使用save()函数保存excel文件 ws.write(0, 0, 1234…...

MYSQL学习笔记(五):单行函数(字符串、数学、日期时间、条件判断、信息、加密、进制转换函数)讲解

前言&#xff1a; 学习和使用数据库可以说是程序员必须具备能力&#xff0c;这里将更新关于MYSQL的使用讲解&#xff0c;大概应该会更新30篇&#xff0c;涵盖入门、进阶、高级(一些原理分析);这一篇是讲解单行函数&#xff0c;当然mysql函数很多哈&#xff0c;只有多用才能记得…...

Grafana系列之Dashboard:新增仪表板、新增变量、过滤变量、变量查询、导入仪表板、变量联动、Grafana Alert

概述 关于Prometheus和Grafana的安装&#xff0c;略过。 写在前面 Dashboard&#xff1a;仪表板&#xff0c;可包含多个PanelPanel&#xff1a;面板&#xff0c;Dashboard中的组件 如有写得不对的地方&#xff0c;烦请指出。 新增仪表板 点击右上角的 选择New dashboard…...

(java版本)基于Misty1算法的加密软件的实现-毕业设计

一、基于Misty1算法的加密软件&#xff08;Java&#xff09;的实现 随着计算机网络及通信技术的飞速发展&#xff0c;信息安全成了信息社会急需解决的最重要的问题之一&#xff0c;密码技术是保证信息安全的核心技术。本文用JAVA语言开发了一个基于Misty1算法的加密软件&#x…...

Spring注解篇:@RestController详解

全文目录&#xff1a; 开篇语前言摘要概述源码解析使用案例分享代码分析使用场景优缺点分析测试用例 应用场景案例优缺点分析核心类方法介绍测试用例测试用例分析使用场景优缺点分析测试用例 小结总结文末 开篇语 哈喽&#xff0c;各位小伙伴们&#xff0c;你们好呀&#xff0c…...

力扣热门100题之最大子数组和

最优解法&#xff1a;Kadane 算法&#xff08;一遍遍历&#xff09;核心思想遍历数组时&#xff0c;维护当前最大和如果前面的和是负数&#xff0c;只会拖累当前数&#xff0c;直接抛弃前面否则&#xff0c;把当前数加进去全程记录最大值完整代码实现&#xff1a;class Solutio…...

蒙特卡洛方法与科学计算十大经典算法解析

1. 蒙特卡洛方法&#xff1a;从赌场到科学计算的跨界革命 1946年&#xff0c;三位天才科学家在洛斯阿拉莫斯实验室的咖啡时间里&#xff0c;可能不会想到他们正在创造一种将彻底改变科学计算的方法。蒙特卡洛方法的名字来源于摩纳哥著名的赌城&#xff0c;这暗示了其核心思想—…...

别再自己造轮子了!用InsightFace+FastAPI快速搭建一个高精度人脸识别Web服务

基于InsightFace的高精度人脸识别Web服务实战指南 1. 为什么选择InsightFace构建人脸识别服务 三年前我接手一个考勤系统项目时&#xff0c;曾花费两个月时间从零训练FaceNet模型&#xff0c;结果上线后遇到戴眼镜员工识别率骤降的问题。直到发现InsightFace这个开源项目&#…...

计算机毕业设计:Python二手车可视化平台 Django框架 可视化 线性回归 数据分析 机器学习 深度学习 AI 大模型(建议收藏)✅

1、项目介绍 技术栈 Python语言、Django框架、ECharts可视化库、机器学习线性回归预测算法、HTML、训练集与测试集划分、模型评估&#xff08;均方误差MSE&#xff09; 功能模块数据可视化分析大屏城市和车型分析里程与价格分析上牌日期分析和颜色分析词云图分析数…...

从电机到USB:一文搞懂嵌入式里的感性负载、容性负载与阻抗匹配(附功率因数校正实例)

从电机到USB&#xff1a;一文搞懂嵌入式里的感性负载、容性负载与阻抗匹配&#xff08;附功率因数校正实例&#xff09; 在嵌入式系统设计中&#xff0c;工程师常常需要同时面对两种看似截然不同的挑战&#xff1a;大功率电机驱动的强电控制和高速数字通信的弱电信号处理。当电…...

如何高效提取Unity游戏资源:AssetStudio的完整实战指南

如何高效提取Unity游戏资源&#xff1a;AssetStudio的完整实战指南 【免费下载链接】AssetStudio AssetStudio - Based on the archived Perfares AssetStudio, I continue Perfares work to keep AssetStudio up-to-date, with support for new Unity versions and additional…...

EViews实战:时间序列分析的平稳性检验与建模全流程

1. 时间序列分析入门&#xff1a;为什么需要平稳性检验&#xff1f; 我第一次接触时间序列分析时&#xff0c;最困惑的就是为什么要做平稳性检验。直到在金融数据分析项目里踩了坑才明白&#xff1a;非平稳序列直接建模会导致预测结果完全失真。比如分析某上市公司股价时&#…...

Python从入门到精通(第17章):继承与多态——is-a关系彻底理解

开头导语 这是本系列第17章。继承和多态是面向对象最核心的概念,但也是被误解最多的概念。继承的本质是“复用”,而不是“is-a”的语言描述;多态的本质是“同一接口,不同实现”,而不是“子类Override父类方法”这个动作本身。本章会从实际场景出发,讲清楚继承的适用边界…...

【无标题】视频号下载神器来了!可指定视频下载,支持批量解析下载

我用夸克网盘给你分享了「链接&#xff1a;https://pan.quark.cn/s/46da937e05b8支持下载指定视频支持批量下载视频支持下载直播视频支持识别已经下载过的视频&#xff0c;不重复下载...

YOLOv13开箱体验:无需配置,直接运行,效果惊艳的目标检测

YOLOv13开箱体验&#xff1a;无需配置&#xff0c;直接运行&#xff0c;效果惊艳的目标检测 1. 从零开始的极速体验 作为一名长期奋战在目标检测一线的开发者&#xff0c;当我第一次接触YOLOv13官版镜像时&#xff0c;最直观的感受就是"快"。这种快不仅体现在模型推…...