软件架构风格系列(4):事件驱动架构
文章目录
- 前言
- 一、从“用户下单”场景看懂事件驱动核心概念
- (一)什么是事件驱动架构?
- (二)核心优势:解耦与异步的双重魔法
- 二、架构设计图:三要素构建事件流转闭环
- 三、Java实战:从简单事件总线到分布式消息队列
- (一)基于Spring Event的轻量级实现(单体应用)
- 1. 定义事件类
- 2. 事件发布者(订单服务)
- 3. 事件订阅者(库存服务)
- (二)分布式场景:基于RabbitMQ的事件驱动(微服务)
- 1. 添加依赖
- 2. 配置队列和交换机
- 3. 生产者发送事件
- 4. 消费者监听事件
- 四、适用场景与避坑指南
- (一)这些场景请优先选择EDA
- (二)踩坑预警:这3个陷阱别掉进去
- 五、总结:事件驱动架构的“成人世界法则”
前言
在电商大促的凌晨零点,当千万用户同时下单时,你是否好奇过系统如何在毫秒级内完成库存扣减、支付通知、物流调度等一系列操作?答案往往藏在一种低调却强大的架构风格里——事件驱动架构(EDA)。作为一个见证过多个大流量系统落地的老湿机,今天就来拆解这种架构的核心逻辑,并用Java代码带你从理论走到实战。
一、从“用户下单”场景看懂事件驱动核心概念
(一)什么是事件驱动架构?
简单来说,它是一种“以事件为中心”的设计模式:
- 事件(Event):系统中发生的“有意义的状态变化”,比如“订单创建”“库存变更”“支付成功”。
- 发布者(Publisher):产生事件的组件(如订单服务),只负责“说发生了什么”,不关心谁来处理。
- 订阅者(Subscriber):对特定事件感兴趣的组件(如库存服务、物流服务),只关注“我该做什么”。
- 事件通道(Event Channel):连接发布者和订阅者的“中介”,可以是内存中的事件总线,也可以是Kafka、RabbitMQ等消息队列。
(二)核心优势:解耦与异步的双重魔法
想象一下传统的“同步调用”:订单服务需要依次调用库存服务扣减库存、支付服务发起扣款、物流服务生成运单,任何一个环节卡顿都会拖慢整个流程。
而事件驱动架构下:
- 订单服务只需发布“订单创建事件”到消息队列,无需等待下游响应;
- 库存、支付、物流服务各自监听队列,异步处理事件;
- 新增功能(如积分系统)只需订阅事件,无需修改原有代码。
这种“发布-订阅”模式让系统像乐高积木一样灵活组装,扛住流量洪峰的同时,还能快速响应业务变化。
二、架构设计图:三要素构建事件流转闭环
- 生产者层:业务触发时生成事件,比如用户提交订单后,订单服务生成
OrderCreatedEvent
。 - 事件通道层:负责事件的可靠传输,支持异步解耦和流量削峰。常见实现包括:
- 轻量级:Spring ApplicationEvent(适合单体应用)
- 分布式:Kafka(适合高吞吐量)、RabbitMQ(适合精准投递)
- 消费者层:监听特定事件并执行业务逻辑,支持横向扩展(如部署多个库存服务实例)。
三、Java实战:从简单事件总线到分布式消息队列
(一)基于Spring Event的轻量级实现(单体应用)
1. 定义事件类
@Getter
public class OrderCreatedEvent {private final String orderId;private final List<String> productIds;public OrderCreatedEvent(String orderId, List<String> productIds) {this.orderId = orderId;this.productIds = productIds;}
}
2. 事件发布者(订单服务)
@Service
public class OrderService {private final ApplicationEventPublisher eventPublisher;@Autowiredpublic OrderService(ApplicationEventPublisher eventPublisher) {this.eventPublisher = eventPublisher;}public String createOrder(String orderId, List<String> productIds) {// 业务逻辑:创建订单记录System.out.println("订单 " + orderId + " 创建成功");// 发布事件eventPublisher.publishEvent(new OrderCreatedEvent(orderId, productIds));return orderId;}
}
3. 事件订阅者(库存服务)
@Service
public class InventoryService {@EventListenerpublic void handleOrderCreated(OrderCreatedEvent event) {String orderId = event.getOrderId();List<String> productIds = event.getProductIds();// 业务逻辑:扣减库存System.out.println("订单 " + orderId + " 触发库存扣减,商品ID:" + productIds);// 模拟库存扣减耗时操作try {Thread.sleep(100);} catch (InterruptedException e) {Thread.currentThread().interrupt();}}
}
(二)分布式场景:基于RabbitMQ的事件驱动(微服务)
1. 添加依赖
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-amqp</artifactId>
</dependency>
2. 配置队列和交换机
@Configuration
public class RabbitMQConfig {public static final String ORDER_QUEUE = "order_queue";public static final String ORDER_EXCHANGE = "order_exchange";public static final String ORDER_ROUTING_KEY = "order.routing.key";@Beanpublic Queue orderQueue() {// 持久化队列return new Queue(ORDER_QUEUE, true); }@Beanpublic DirectExchange orderExchange() {return new DirectExchange(ORDER_EXCHANGE);}@Beanpublic Binding orderBinding(Queue orderQueue, DirectExchange orderExchange) {return BindingBuilder.bind(orderQueue).to(orderExchange).with(ORDER_ROUTING_KEY);}
}
3. 生产者发送事件
@Service
public class RabbitMQProducer {private final RabbitTemplate rabbitTemplate;@Autowiredpublic RabbitMQProducer(RabbitTemplate rabbitTemplate) {this.rabbitTemplate = rabbitTemplate;}public void sendOrderEvent(OrderCreatedEvent event) {// 将事件序列化为JSON发送到队列rabbitTemplate.convertAndSend(RabbitMQConfig.ORDER_EXCHANGE,RabbitMQConfig.ORDER_ROUTING_KEY,event);}
}
4. 消费者监听事件
@Service
public class RabbitMQConsumer {@RabbitListener(queues = RabbitMQConfig.ORDER_QUEUE)public void receiveOrderEvent(OrderCreatedEvent event) {// 处理逻辑与单体应用类似,支持分布式部署System.out.println("分布式场景接收到事件:" + event.getOrderId());}
}
四、适用场景与避坑指南
(一)这些场景请优先选择EDA
- 高并发异步处理:电商下单、秒杀活动,通过事件队列削峰填谷,避免数据库被压垮。
- 微服务解耦:不同微服务通过事件通信,服务A无需知道服务B的接口和地址,降低耦合度。
- 实时数据处理:金融行情分析、用户行为追踪,事件流处理框架(如Kafka Streams)可实时消费事件并计算。
- 最终一致性:分布式事务中,通过事件重试和补偿机制实现最终一致性(如TCC模式结合事件驱动)。
(二)踩坑预警:这3个陷阱别掉进去
- 事件膨胀:避免在事件中携带过多数据(如整个订单对象),只传递必要字段(如
orderId
),减少网络传输开销。 - 顺序性问题:对订单支付、退款等有严格顺序的事件,需在事件中添加
sequenceId
,消费者按顺序处理(可借助Redis实现分布式锁)。 - 事件回溯困难:生产环境建议使用可持久化的消息队列,并记录事件日志,方便故障时重放事件恢复状态。
五、总结:事件驱动架构的“成人世界法则”
事件驱动架构的本质,是承认系统的“不完美”:
- 接受组件可能崩溃,通过事件重试和幂等性设计保证可靠性;
- 接受需求会变化,通过松耦合让系统像搭积木一样灵活组装;
- 接受流量会波动,通过异步队列将突发压力转化为可处理的平稳水流。
从单体应用的Spring Event到分布式系统的Kafka,这种架构风格贯穿了从小型工具到亿级流量平台的全生命周期。如果你正在设计一个需要“抗住现在、适配未来”的系统,事件驱动架构绝对是值得深入研究的“秘密武器”。
最后留个小问题:你认为事件驱动架构和微服务架构是如何相辅相成的?欢迎在评论区聊聊你的想法~
图片来源于网络
相关文章:

软件架构风格系列(4):事件驱动架构
文章目录 前言一、从“用户下单”场景看懂事件驱动核心概念(一)什么是事件驱动架构?(二)核心优势:解耦与异步的双重魔法 二、架构设计图:三要素构建事件流转闭环三、Java实战:从简单…...
windows系统各版本下载
以下各版本Windows系统链接来自网友整理,请通过迅雷或者其他支持ED2K或BT的下载工具进行下载。 注:以下为原版系统,未激活、非破解版,仅供下载体验学习,请勿从事商业活动。 Windows 11 Windows 11 (consumer editions…...

arduino平台读取鼠标光电传感器
鼠标坏掉了,大抵是修不好了。(全剧终—) 但是爱动手的小明不会浪费这个鼠标,确认外观没有明显烧毁痕迹后,尝试从电路板上利用光电传感器进行位移的测量,光电传感器(型号:FCT3065&am…...

【Linux网络】网络层
网络层 在复杂的网络环境中确定一个合适的路径 IP 协议 IPV4 点分十进制[0,255].[0,255].[0,255].[0,255]IPV6 IP地址目标网格目标主机 基本概念 主机:配有IP地址,但是不进行路由控制的设备;路由器:即配有IP地址,又能进行路由控制;节点:主机和路由器的统称。 两个问题 路…...
力扣-98.验证二叉搜索树
题目描述 给你一个二叉树的根节点 root ,判断其是否是一个有效的二叉搜索树。 有效 二叉搜索树定义如下: 节点的左子树只包含 小于 当前节点的数。节点的右子树只包含 大于 当前节点的数。所有左子树和右子树自身必须也是二叉搜索树。 class Solutio…...
5.17本日总结
一、英语 复习list2list29 二、数学 学习14讲部分内容 三、408 学习计组1.2内容 四、总结 高数和计网明天结束当前章节,计网内容学完之后主要学习计组和操作系统 五、明日计划 英语:复习lsit3list28,完成07年第二篇阅读 数学&#…...

大模型学习:Deepseek+dify零成本部署本地运行实用教程(超级详细!建议收藏)
文章目录 大模型学习:Deepseekdify零成本部署本地运行实用教程(超级详细!建议收藏)一、Dify是什么二、Dify的安装部署1. 官网体验2. 本地部署2.1 linux环境下的Docker安装2.2 Windows环境下安装部署DockerDeskTop2.3启用虚拟机平台…...
VSCode launch.json 配置参数详解
使用 launch.json 配置调试环境时,会涉及到多个参数,用于定义调试器的行为和目标执行环境。以下是一些常用的配置参数: 1、"type" :指定调试器的类型,例如 "node" 表示 Node.js 调试器࿰…...
pytest多种断言类型封装为自动化断言规则库
以下是将多种断言类型封装为自动化断言规则库的完整实现方案,包含基础验证规则和扩展机制: import re import time from jsonschema import validate, ValidationError from typing import Dict, Any, Optional, Callableclass ResponseValidator:"""自动...
Oracle数据库如何进行冷备份和恢复
数据库的冷备份指的是数据库处于关闭或者MOUNT状态下的备份,备份文件包括数据文件、日志文件和控制文件。数据库冷备份所用的时间主要受数据库大小和磁盘I/O性能的影响。由于数据库需要关闭才能进行冷备份,所以这种备份技术并不适用724小时的系统。尽管冷…...

LeetCode Hot100 (2、3、4、5、6、8、9、12)
题2--字母异或位分词 class Solution { public:vector<vector<string>> groupAnagrams(vector<string>& strs) {// 一开始的思路是,对于其中的一个单词,遍历所有排序组合,然后判断这些组合是否在哈希表里//࿰…...

FastMCP:为大语言模型构建强大的上下文和工具服务
FastMCP:为大语言模型构建强大的上下文和工具服务 在人工智能快速发展的今天,大语言模型(LLM)已经成为许多应用的核心。然而,如何让这些模型更好地与外部世界交互,获取实时信息,执行特定任务&am…...

数据结构(3)线性表-链表-单链表
我们学习过顺序表时,一旦对头部或中间的数据进行处理,由于物理结构的连续性,为了不覆盖,都得移,就导致时间复杂度为O(n),还有一个潜在的问题就是扩容,假如我们扩容前是10…...

Java Solon v3.3.0 发布(国产优秀应用开发基座)
Solon 框架! Solon 是新一代,Java 企业级应用开发框架。从零开始构建(No Java-EE),有灵活的接口规范与开放生态。采用商用友好的 Apache 2.0 开源协议,是“杭州无耳科技有限公司”开源的根级项目ÿ…...

23种设计模式概述详述(C#代码示例)
文章目录 1. 引言1.1 设计模式的价值1.2 设计模式的分类 2. 面向对象设计原则2.1 单一职责原则 (SRP)2.2 开放封闭原则 (OCP)2.3 里氏替换原则 (LSP)2.4 接口隔离原则 (ISP)2.5 依赖倒置原则 (DIP)2.6 合成复用原则 (CRP)2.7 迪米特法则 (LoD) 3. 创建型设计模式3.1 单例模式 (…...

数字化工厂升级引擎:Modbus TCP转Profinet网关助力打造柔性生产系统
在当今的工业自动化领域,通信协议扮演着至关重要的角色。Modbus TCP和Profinet是两种广泛使用的工业通信协议,它们分别在不同的应用场景中发挥着重要作用。然而,有时我们可能需要将这两种协议进行转换,以实现不同设备之间的无缝通…...

FPGA生成随机数的方法
FPGA生成随机数的方法,目前有以下几种: 1、震荡采样法 实现方式一:通过低频时钟作为D触发器的时钟输入端,高频时钟作为D触发器的数据输入端,使用高频采样低频,利用亚稳态输出随机数。 实现方式二:使用三个…...

【Linux C/C++开发】轻量级关系型数据库SQLite开发(包含性能测试代码)
前言 之前的文件分享过基于内存的STL缓存、环形缓冲区,以及基于文件的队列缓存mqueue、hash存储、向量库annoy存储,这两种属于比较原始且高效的方式。 那么,有没有高级且高效的方式呢。有的,从数据角度上看,࿰…...
2025认证杯第二阶段数学建模B题:谣言在社交网络上的传播思路+模型+代码
2025认证杯数学建模第二阶段思路模型代码,详细内容见文末名片 一、引言 在当今数字化时代,社交网络已然成为人们生活中不可或缺的一部分。信息在社交网络上的传播速度犹如闪电,瞬间就能触及大量用户。然而,这也为谣言的滋生和扩…...

记录算法笔记(2025.5.17)验证二叉搜索树
给你一个二叉树的根节点 root ,判断其是否是一个有效的二叉搜索树。 有效 二叉搜索树定义如下: 节点的左子树只包含 小于 当前节点的数。节点的右子树只包含 大于 当前节点的数。所有左子树和右子树自身必须也是二叉搜索树。 示例 1: 输入&…...

flutter编译时 设置jdk版本
先查看flutter使用的版本 flutter doctor -v设置flutter的jdk目录 flutter config --jdk-dir "E:\soft\android-studio\jbr" 然后再验证下,看是否设置成功...

ctfshow——web入门254~258
目录 web入门254 web入门255 web入门256 web入门257 web入门258 反序列化 先来看看其他师傅的讲解 web入门254 源码: <?phperror_reporting(0); highlight_file(__FILE__); include(flag.php);class ctfShowUser{public $usernamexxxxxx;public $passwo…...

【数据处理】xarray 数据处理教程:从入门到精通
目录 xarray 数据处理教程:从入门到精通一、简介**核心优势** 二、安装与导入1. 安装2. 导入库 三、数据结构(一)DataArray(二) Dataset(三)关键说明 四、数据操作(一)索…...

qt5.14.2 opencv调用摄像头显示在label
ui界面添加一个Qlabel名字是默认的label 还有一个button名字是pushButton mainwindow.h #ifndef MAINWINDOW_H #define MAINWINDOW_H#include <QMainWindow> #include <opencv2/opencv.hpp> // 添加OpenCV头文件 #include <QTimer> // 添加定…...
科技的成就(六十八)
623、杰文斯悖论 杰文斯悖论是1865年经济学家威廉斯坦利杰文斯提出的一悖论:当技术进步提高了效率,资源消耗不仅没有减少,反而激增。例如,瓦特改良的蒸汽机让煤炭燃烧更加高效,但结果却是煤炭需求飙升。 624、代码混…...

芯片生态链深度解析(三):芯片设计篇——数字文明的造物主战争
【开篇:设计——数字文明的“造物主战场”】 当英伟达的H100芯片以576TB/s显存带宽重构AI算力边界,当阿里平头哥倚天710以RISC-V架构实现性能对标ARM的突破,这场围绕芯片设计的全球竞赛早已超越技术本身,成为算法、架构与生态标准…...

Rocky Linux 9.5 基于kubeadm部署k8s
一:部署说明 操作系统https://mirrors.aliyun.com/rockylinux/9.5/isos/x86_64/Rocky-9.5-x86_64-minimal.iso 主机名IP地址配置k8s- master192.168.1.1412颗CPU 4G内存 100G硬盘k8s- node-1192.168.1.1422颗CPU 4G内存 100G硬盘k8s- node-2192.168.1.1432…...
--openssl-legacy-provider is not allowed in NODE_OPTIONS 报错的处理方式
解决方案 Node.js 应用: 从 Node.js v17 开始,底层升级到 OpenSSL 3.0,可能导致旧代码报错(如 ERR_OSSL_EVP_UNSUPPORTED)。 通过以下命令启用旧算法支持: node --openssl-legacy-provider your_script.js…...

uart16550详细说明
一、介绍 uart16550 ip core异步串行通信IP连接高性能的微控制器总线AXI,并为异步串行通信提供了 控制接口。软核设计连接了axilite接口。 二、特性 1.axilite接口用于寄存器访问和数据传输 2.16650串口和16450串口的软件和硬件寄存器都是兼容的 3.默认的core配置参数…...
deepin v23.1 音量自动静音问题解决
有的机器上会有音量自动静音问题, 如果你的电脑上也遇到, 这个问题是 Linux 内核的原因, ubuntu上也可能会遇到相同问题(比如你升级了最新内核6.14), 而我测试得6.8.0的内核是不会自动静音的. Index of /mainline 到上面这个链接(linux 内核的官方链接)下载6.8.0的内核, s…...