一、前言

在单体应用中,事务一般由关系型数据库本身来保证,通过 ACID 特性实现数据一致性。但随着微服务架构的普及,应用被拆分为多个独立服务,数据可能分散在不同数据库、不同存储引擎中,传统的单机事务无法再覆盖。

这就引出了 分布式事务 的问题:如何在多服务、多数据库场景下,仍然保证数据一致性?

本文将结合 Spring Boot 实际开发,对常见的几种分布式事务方案进行解析:

  • XA 方案(两阶段提交,强一致性)

  • Seata(柔性事务,常用于阿里系微服务)

  • 本地消息表 + 最终一致性(高可用场景的常见落地方案)

同时会总结它们的 优缺点、常见坑点与适用场景

二、分布式事务常见场景

在 Spring Boot 开发中,以下场景经常涉及分布式事务:

  1. 订单系统:创建订单 → 扣减库存 → 扣减余额

  2. 支付系统:支付成功 → 修改订单状态 → 发送消息通知 → 更新积分

  3. 营销系统:用户下单 → 触发优惠券核销 → 更新活动数据

这些流程往往跨越多个服务和数据库,如果其中一步失败,就可能导致数据不一致,例如:

  • 库存已扣减,但订单未生成

  • 订单已支付,但未发货

  • 优惠券已核销,但活动未更新

因此需要合理的分布式事务方案来保证一致性。

三、XA 方案(两阶段提交)

1. 原理

XA 是 分布式事务标准协议,基于两阶段提交(2PC,Two Phase Commit):

  • 阶段一:事务协调者向所有数据库发送 prepare,各数据库执行但不提交,返回可提交状态

  • 阶段二:如果所有数据库都返回成功,则发送 commit,否则发送 rollback

2. 在 Spring Boot 中的实现

常见实现方式是 Atomikos、Narayana 等第三方事务管理器,也可以结合 JTA 来管理。

示例配置(Atomikos):

spring:jta:enabled: trueatomikos:properties:service: com.atomikos.icatch.standalone.UserTransactionServiceFactory

3. 优缺点

优点

  • 强一致性保证

  • 对开发透明,业务代码几乎不用改

缺点

  • 性能差:两阶段提交会增加锁时间,导致吞吐量下降

  • 扩展性差:跨多数据源时容易成为瓶颈

  • 单点风险:协调者挂掉可能导致事务悬挂

4. 适用场景

适合金融级场景(如银行转账)——必须保证强一致性,但对性能要求相对次要。

四、Seata(柔性事务)

1. 原理

Seata 是阿里开源的分布式事务解决方案,支持 AT 模式、TCC 模式、Saga 模式

  • AT 模式:自动代理数据源,类似本地事务 + Undo Log 回滚,开发成本低

  • TCC 模式:需要业务实现 Try/Confirm/Cancel 三个接口,粒度更细,性能更高

  • Saga 模式:长事务补偿,适合跨服务调用链较长的业务

2. 在 Spring Boot 中集成

依赖:

<dependency><groupId>io.seata</groupId><artifactId>seata-spring-boot-starter</artifactId><version>1.7.1</version>
</dependency>

配置:

seata:enabled: trueapplication-id: order-servicetx-service-group: my_tx_groupservice:vgroup-mapping:my_tx_group: default

使用:

@GlobalTransactional
public void createOrder(Order order) {orderDao.save(order);inventoryService.deduct(order.getId());
}

3. 优缺点

优点

  • 透明代理数据库,AT 模式对开发友好

  • 多种事务模式可选,适配不同业务场景

  • 社区活跃,生态完善

缺点

  • 需要额外部署 Seata Server

  • Undo Log 占用空间,长事务性能下降

  • 某些复杂 SQL(批量更新/存储过程)支持有限

4. 适用场景

电商下单、库存扣减等典型 高并发场景,对性能要求较高,能接受短时间的不一致。

五、本地消息表(最终一致性)

1. 原理

核心思想是 业务数据 + 消息发送放在同一个本地事务中,通过消息队列来保证最终一致性:

  1. 业务服务在本地事务中写入业务表 + 消息表

  2. 定时任务扫描消息表,发送消息到 MQ

  3. 消费者消费 MQ 消息,执行后续逻辑

  4. 成功后更新消息表状态

2. 在 Spring Boot 中实现

数据库表:

CREATE TABLE t_outbox_message (id BIGINT PRIMARY KEY AUTO_INCREMENT,content TEXT NOT NULL,status TINYINT DEFAULT 0,create_time TIMESTAMP DEFAULT CURRENT_TIMESTAMP
);

业务代码:

@Transactional
public void createOrder(Order order) {orderDao.insert(order);outboxDao.insert(new OutboxMessage(order));
}

定时任务发送消息:

@Scheduled(fixedRate = 5000)
public void sendPendingMessages() {List<OutboxMessage> messages = outboxDao.findPending();for (OutboxMessage msg : messages) {mqProducer.send(msg);outboxDao.markSent(msg.getId());}
}

3. 优缺点

优点

  • 无需额外中间件,简单可靠

  • 高可用,适合最终一致性场景

缺点

  • 开发成本较高,需要写消息表逻辑

  • 延迟一致性(可能有几秒的延迟)

  • 消息补偿、幂等处理逻辑复杂

4. 适用场景

订单系统、积分系统、日志收集等对 最终一致性容忍度高 的业务。

六、三种方案对比

方案一致性性能开发成本典型场景
XA强一致银行转账、核心金融业务
Seata最终一致(AT/TCC)中高电商下单、库存、支付
本地消息表最终一致营销、积分、日志系统

七、常见坑点总结

  1. XA 事务悬挂:协调者挂掉可能导致数据库锁未释放

  2. Seata Undo Log 膨胀:要定期清理,否则影响性能

  3. 本地消息表补偿失败:要考虑消息幂等、死信队列机制

八、结语

分布式事务没有“银弹”,不同方案适合的场景完全不同:

  • 金融强一致 → 优先考虑 XA

  • 电商高并发 → 选择 Seata AT/TCC

  • 最终一致性即可 → 本地消息表是首选

在 Spring Boot 实践中,建议结合 业务场景 + 性能要求 + 容错能力 来选择合适的方案,而不是盲目套用。


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/diannao/98844.shtml
繁体地址,请注明出处:http://hk.pswp.cn/diannao/98844.shtml
英文地址,请注明出处:http://en.pswp.cn/diannao/98844.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Transporter App 使用全流程详解:iOS 应用 ipa 上传工具、 uni-app 应用发布指南

在 iOS 应用开发与发布过程中&#xff0c;Transporter App 是苹果官方提供的一款上传工具&#xff0c;专门用于将 ipa 文件 或 应用资源 上传到 App Store Connect。 与 Xcode 上传 相比&#xff0c;Transporter 更加稳定&#xff0c;尤其适合大文件上传&#xff0c;因此在 iOS…

计算机毕业设计 基于Hadoop的B站数据分析可视化系统的设计与实现 Python 大数据毕业设计 Hadoop毕业设计选题【附源码+文档报告+安装调试】

博主介绍&#xff1a;✌从事软件开发10年之余&#xff0c;专注于Java技术领域、Python、大数据、人工智能及数据挖掘、小程序项目开发和Android项目开发等。CSDN、掘金、华为云、InfoQ、阿里云等平台优质作者✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&…

使用自定义LLM和Embedding模型部署Vanna:基于RAG的Text-to-SQL生成

使用自定义LLM和Embedding模型部署Vanna&#xff1a;基于RAG的Text-to-SQL生成 说明&#xff1a; 首次发表日期&#xff1a;2024-07-12Vanna Github地址&#xff1a; https://github.com/vanna-ai/vannaVanna官方文档&#xff1a; https://vanna.ai/ 部署Vanna时我们可以选择使…

Linux多线程概念

背景知识内存管理OS进行内存管理不是以字节为单位的&#xff0c;而是以内存块为单位的&#xff0c;默认大小为4kb&#xff1b;系统和磁盘文件进行IO交互的单位是4kb&#xff08;8个扇区&#xff09;&#xff1b;OS对内存管理实质上是对页框进行管理。页框&#xff08;Page Fram…

【Problem】动态规划之跳跃游戏系列

一、跳跃游戏 55. 跳跃游戏 - 力扣&#xff08;LeetCode&#xff09;https://leetcode.cn/problems/jump-game/description/?envTypeproblem-list-v2&envIddynamic-programming class Solution { public:bool canJump(vector<int>& nums) {// 状态定义&#x…

射频EVM

EVM&#xff08;Error Vector Magnitude&#xff0c;误差矢量幅度&#xff09;是衡量无线通信系统中调制质量的重要指标&#xff0c;尤其用于评估信号的调制误差和系统性能。它通常用来表示传输信号与理想信号之间的偏差&#xff0c;特别是在数字通信中。EVM的基本概念&#xf…

Java 更改 Word 文档中文本颜色

在日常的自动化文档处理中&#xff0c;我们经常会遇到需要对 Word 文档内容进行编程修改的需求&#xff0c;其中一项常见且重要的操作就是更改文本的颜色。无论是为了突出重点、统一品牌风格&#xff0c;还是实现动态内容展示&#xff0c;精准地修改文本颜色都是一个核心痛点。…

STM32—SPI协议

文章目录一、SPI 协议简介二、硬件电路2.1.SPI的连接2.2.数据的移位2.3.时序基本单元2.3.1.起始条件和终止条件2.3.2.模式 02.3.3.模式 12.3.4.模式 22.3.5.模式 32.4.时序三、软件实现四、W25Q644.1.简介4.2.硬件电路4.3.框图4.4.操作注意事项五、实验一、SPI 协议简介 SPI&a…

Qt中的QWebEngineView

第1章 本地目录结构1.1 自己写的两个网页(html)mermaid.html &#xff08;自己写的网页界面&#xff09;WebTest.html (自己写的网页界面)qwebchannel.js (Qt下载安装之后&#xff0c;会在安装目录下有这个文件&#xff0c;需要将安装目录下的改文件拷贝…

Flutter 应用国际化 (i18n) 与本地化 (l10n) 完整指南

Flutter 国际化 (i18n) 完全指南&#xff1a;从入门到精通 在现代移动应用开发中&#xff0c;支持多语言是触达全球用户的基本要求。Flutter 提供了强大且灵活的国际化 (i18n) 和本地化 (l10n) 支持。本文将带你从零开始&#xff0c;一步步深入掌握在 Flutter 中实现国际化的几…

计算机视觉与深度学习 | 计算机视觉中线特征提取与匹配算法综述

文章目录 一、线特征提取算法原理 1.1 Hough变换及其优化 1.2 LSD算法 1.3 EDLines算法 二、核心数学公式 2.1 直线表示与误差计算 2.2 LSD算法关键公式 三、线特征匹配算法 3.1 LBD描述符 3.2 匹配策略 四、代码实现 4.1 LSD线段检测(Python) 4.2 LBD特征匹配(C++) 五、算…

Transformer 模型:Attention is All You Need 的真正含义

2017 年&#xff0c;Google Brain 发布了一篇具有里程碑意义的论文——《Attention Is All You Need》&#xff0c;这篇论文不仅首次提出了 Transformer 模型&#xff0c;更重要的是&#xff0c;它宣称“注意机制&#xff08;Attention Mechanism&#xff09;就足以构建强大的模…

数据库约束表的设计

数据库约束概念&#xff1a;数据库约束是关系型数据库的一个重要功能&#xff0c;主要是保证数据的完整性&#xff0c;也可理解为数据的正确性&#xff08;数据本身是否正确&#xff0c;关联关系是否正确&#xff09;&#xff08;一般是用在指定列上&#xff09;常见的约束类型…

【案例分享】TeeChart 助力 Softdrill 提升油气钻井数据可视化能力

在钻井与地质工程领域&#xff0c;数据可视化是核心环节。图表不仅需要精确与高效&#xff0c;还需符合行业习惯并支持交互与定制。Softdrill 自 2012 年起在核心产品中集成了TeeChart 图表库&#xff0c;将复杂的井下数据转化为直观的工程图表&#xff0c;极大提升了钻井工程师…

【Flink】Flink Runtime 架构设计

Flink Runtime 架构设计 整体架构 ┌─────────────────────────────────────────────────────────────────┐ │ Flink Runtime │ ├─────────…

Git 命令教程

Git介绍 分布式版本控制系统。 Git命令 初始化/全局配置git init初始化一个Git仓库&#xff08;会创建一个.git的目录&#xff09;git config --global user.name “name”设置提交时的用户名git config user.name查看设置的用户名git config --global user.email “youemail.c…

git config --global user.name指令报错时的解决方案

问题分析 %HOMEDRIVE%%HOMEPATH%/.gitconfig 是Windows环境变量的表示方式&#xff1a; %HOMEDRIVE% 通常是 C:%HOMEPATH% 通常是 \Users\你的用户名完整路径应该是&#xff1a;C:\Users\你的用户名\.gitconfig 但这里环境变量没有被正确解析&#xff0c;显示的是字面意思。 …

websocket和socket io的区别

好的&#xff0c;这是一个更具体也更常见的问题。WebSocket 是一种协议&#xff0c;而 Socket.IO 是一个库&#xff0c;它使用了 WebSocket 但提供了多得多的功能。 简单比喻&#xff1a; WebSocket 就像是给你提供了一条高效的“快递专线”&#xff08;双向通信通道&#xff…

Nginx反向代理与负载均衡部署

Nginx反向代理与负载均衡部署实战指南前言一、规划部署负载均衡和反向代理二、部署Nginx负载均衡器2.1. 准备基础环境2.2. 创建Nginx运行用户2.3. 编译安装Nginx2.4. 配置Nginx系统服务2.5. 验证Nginx安装三、部署后端2台Tomcat应用服务器3.1. 安装JDK3.2. 部署Tomcat实例13.3.…

从源码和设计模式深挖AQS(AbstractQueuedSynchronizer)

AQS 概念 AbstractQueuedSynchronizer&#xff08;AQS&#xff09; 是 Java 并发包 (java.util.concurrent.locks) 的核心基础框架&#xff0c;它的实现关键是先进先出 (FIFO) 等待队列和一个用volatile修饰的锁状态status。具体实现有 : ReentrantLock、Semaphore、CountDownL…