news 2026/4/24 4:53:25

经典场景设计方案系列---【分布式事务】

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
经典场景设计方案系列---【分布式事务】

1.场景

如何保证“本地数据库插入”与“调用第三方接口”这两个操作的原子性(要么都成功,要么都失败),这是一个非常经典且常见的分布式事务场景。

2.方案一:调整顺序 + 本地事务(适用于轻量级、对即时性要求不高的场景)

这是最简单且最推荐的方案,核心思想是先落库,再同步

  • 1.逻辑设计:

    • 1.在本地数据库表中增加一个状态字段(例如 sync_status:0-未同步,1-已同步)。
    • 2.开启本地事务 @Transactional。
    • 3.先将数据插入本地数据库,标记状态为“未同步”。
    • 4.提交本地事务(此时数据已入库,但未同步)。
    • 5.在事务提交后(或通过异步线程/消息队列)调用第三方接口。
    • 6.如果第三方调用成功,更新本地数据库状态为“已同步”。
  • 2.异常处理(补偿机制):

    • 定时任务: 编写一个定时任务(Scheduled Task),定期扫描数据库中状态为“未同步”且创建时间超过一定阈值的数据,重新发起同步请求。
    • 优点: 即使第三方接口挂了,本地数据也不会丢失,保证了最终一致性。
    • 缺点: 存在短暂的数据不一致。

示例代码:

// 1. 本地保存(事务内) @Transactional(rollbackFor = Exception.class) public void saveLocal(Data data) { data.setSyncStatus("UN_SYNC"); mapper.insert(data); } // 2. 主业务逻辑 public void saveAndSync(Data data) { // 第一步:先落库(保证本地数据安全) saveLocal(data); // 第二步:尝试同步(即使这里失败了,也不会回滚上面的saveLocal) try { boolean result = thirdPartyClient.call(data); if (result) { // 同步成功,更新状态 mapper.updateStatus(data.getId(), "SYNCED"); } } catch (Exception e) { log.error("同步失败,等待定时任务补偿", e); // 这里吞掉异常,不要影响主流程返回成功 } } // 3. 另外编写一个定时任务 (例如每5分钟执行一次) @Scheduled(cron = "0 0/5 * * * ?") public void retrySyncTask() { List<Data> pendingData = mapper.selectUnSyncData(); for (Data data : pendingData) { // 重新调用第三方,成功后更新状态 // 建议设置最大重试次数,超过后人工介入 } }

3.方案二:利用消息队列(MQ)实现最终一致性(适用于高并发场景)

如果您的系统已经引入了消息队列(如 RabbitMQ, RocketMQ, Kafka),可以使用可靠消息服务。

  • 1.逻辑设计:

    • 本地事务内: 插入业务数据,同时插入一条“待发送消息”记录到一张本地消息表(Local Message Table),这两步在同一个数据库事务中,保证百分百成功。
    • 异步发送: 这一步有多种实现方式:
      • 方式A(轮询): 定时任务扫描本地消息表,投递到 MQ。
      • 方式B(事务消息): 如果使用 RocketMQ,可以直接利用其“事务消息”特性。
    • 消费端: 消费者监听 MQ,收到消息后调用第三方接口。
    • 确认机制: 只有第三方接口调用成功,才确认消费消息(ACK);如果失败,MQ 会重试。
  • 2.优点: 解耦了业务逻辑和第三方调用,系统吞吐量高。

  • 3.注意: 消费者端需要做好幂等性处理(防止第三方接口被重复调用)。

示例代码:

1.核心业务类(生产端)

这里是关键:业务数据入库和本地消息入库必须在同一个 @Transactional 事务中。

@Service public class UserService { @Autowired private UserMapper userMapper; @Autowired private LocalMessageMapper messageMapper; @Autowired private RabbitMQService rabbitMQService; // 封装的MQ发送服务 /** * 注册用户并触发同步 */ @Transactional(rollbackFor = Exception.class) // 开启本地事务 public void registerUser(User user) { // 1. 插入业务数据 userMapper.insert(user); // 2. 组装消息内容 String msgId = UUID.randomUUID().toString(); UserSyncMsg msgContent = new UserSyncMsg(user.getId(), user.getName()); String json = JSON.toJSONString(msgContent); // 3. 插入本地消息表 (状态为 0-待发送) // 这步非常关键!它保证了如果数据库回滚,消息记录也会回滚;如果提交,消息记录一定存在。 LocalMessage localMsg = new LocalMessage(); localMsg.setId(msgId); localMsg.setMsgContent(json); localMsg.setExchange("user.exchange"); localMsg.setRoutingKey("user.sync.crm"); localMsg.setStatus(0); messageMapper.insert(localMsg); // 4. 发送MQ消息 (这步其实可以异步,或者放在事务提交后的回调中,这里为了简单直接调用) // 注意:即使这里发MQ失败报错,也不要抛出异常导致事务回滚。 // 因为我们有定时任务兜底(见第4步)。 try { rabbitMQService.send(msgId, json); // 发送成功,更新本地消息状态为 1-已发送 messageMapper.updateStatus(msgId, 1); } catch (Exception e) { log.error("MQ发送失败,等待定时任务补偿: {}", msgId); // 这里吞掉异常,不要影响主业务注册成功 } } }

2.消费者监听 (Consumer)

消费者负责调用第三方接口。如果失败,利用 MQ 的重试机制或死信队列。

@Component @RabbitListener(queues = "user.sync.queue") public class UserSyncConsumer { @Autowired private ThirdPartyCrmClient crmClient; @RabbitHandler public void process(String msgJson, Channel channel, @Header(AmqpHeaders.DELIVERY_TAG) long tag) { UserSyncMsg msg = JSON.parseObject(msgJson, UserSyncMsg.class); try { // 1. 幂等性检查 (非常重要!) // 调用三方接口前,先查一下三方或者本地Redis,确保这个userId没有被同步过。 // if (isSynced(msg.getUserId())) { channel.basicAck(tag, false); return; } // 2. 调用第三方接口 boolean success = crmClient.syncUserToCrm(msg); if (success) { // 3. 成功,手动确认消息 (ACK) channel.basicAck(tag, false); log.info("同步第三方成功: {}", msg.getUserId()); } else { // 4. 业务逻辑失败(比如参数校验不过),通常不再重试,或者进入死信队列人工处理 log.error("第三方返回失败"); channel.basicNack(tag, false, false); // 不重回队列,转入死信或丢弃 } } catch (Exception e) { // 5. 网络抖动等异常,拒绝消息并重回队列 (Requeue = true) // 这样MQ过一会会再次推送这条消息 try { // 也可以结合重试次数判断,如果重试太多次就丢进死信队列 channel.basicNack(tag, false, true); } catch (IOException ex) { ex.printStackTrace(); } } } }

3.兜底定时任务 (补偿机制)

这是“最终一致性”的保障。防止第2步中 rabbitMQService.send 失败(例如MQ挂了),导致本地消息表一直是“待发送”状态。

@Component public class MessageResendTask { @Autowired private LocalMessageMapper messageMapper; @Autowired private RabbitMQService rabbitMQService; // 每分钟扫描一次状态为 0 (待发送) 且创建时间超过1分钟的消息 @Scheduled(fixedRate = 60000) public void resendFailedMessages() { List<LocalMessage> failedMsgs = messageMapper.selectPendingMessages(); for (LocalMessage msg : failedMsgs) { if (msg.getRetryCount() > 5) { // 超过最大重试次数,标记为失败,报警人工介入 messageMapper.updateStatus(msg.getId(), 2); continue; } try { rabbitMQService.send(msg.getId(), msg.getMsgContent()); // 发送成功,更新状态 messageMapper.updateStatus(msg.getId(), 1); } catch (Exception e) { // 再次失败,增加重试次数 messageMapper.incrementRetryCount(msg.getId()); } } } }

4.方案总结

这个 Demo 实现了以下逻辑闭环:

  • 1.原子性: 用户数据和消息记录在同一个数据库事务中,同生共死。
  • 2.可靠性: 即使第一遍发 MQ 失败,定时任务会扫描本地消息表进行补发。
  • 3.最终一致性: 消费者拿到消息后,不断重试调用第三方,直到成功(或进入死信队列)。
  • 4.解耦: 注册操作极快,不需要等待第三方接口响应。
    关键点提示:
    消费者幂等性: 第三方接口可能会被重复调用(比如消费者处理完了,提交 ACK 时网络断了,MQ 以为没成功又发了一遍),所以消费者内部或者第三方接口必须能处理重复请求。

4.方案三:最大努力通知(Best Effort Notification)

如果您不想引入复杂的 MQ 或定时任务,可以在代码层面做简单的“重试”。

  • 1.逻辑设计:

    • 开启本地事务。
    • 插入数据库。
    • 注意: 此时不要提交事务。
    • 调用第三方接口。
    • 如果调用成功: 提交本地事务。
    • 如果调用失败: 抛出异常,回滚本地事务(此时本地数据也就没了)。
  • 2.重大缺陷:

    • 长事务问题: 网络请求耗时不可控,会导致数据库连接被长时间占用,严重影响数据库性能。
    • “假失败”问题: 如果第三方接口已经处理成功,但返回响应时网络超时,你的代码会认为失败并回滚本地数据,导致第三方有数据而本地没有,造成严重的数据不一致。
    • 因此: 强烈不建议在 @Transactional 事务代码块内部直接进行 HTTP/RPC 网络请求。

5.方案四:Seata 等分布式事务框架(TCC 模式)

如果业务场景非常严格,要求强一致性(几乎同时成功或失败),可以使用分布式事务框架(如 Alibaba Seata)。

  • 1.逻辑设计(TCC模式):
    • Try: 预留资源(本地数据插入中间状态)。
    • Confirm: 确认提交(本地更新状态,调用第三方接口正式处理)。
    • Cancel: 回滚(删除本地数据,调用第三方接口的“取消/回滚”方法)。
  • 2.难点: 这要求第三方接口必须配合您,提供 Try, Confirm, Cancel 三个配套接口。大多数第三方系统并不支持这种模式。

6.总结建议

综合考虑开发成本和系统稳定性,方案一(本地消息表+定时任务补偿) 是性价比最高的选择。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/23 13:50:55

取一个奶奶辈的微信昵称[特殊字符],好听到爆

&#x1f338; 俏皮日常类 • 广场舞C位&#x1f483;快乐翻倍✨ • 超市薅羊毛&#x1f6d2;满载而归&#x1f36c; • 追剧不停歇&#x1f4fa;零食管够&#x1f36a; • 公园遛弯儿&#x1f6b6;‍♀️偶遇老友&#x1f475; •晒衣晒太阳&#x1f31e;心情亮堂堂☀️ &…

作者头像 李华
网站建设 2026/4/20 17:19:15

生日祝福语音定制服务商业模式探讨

生日祝福语音定制服务的商业实践与技术融合 在某个深夜&#xff0c;一位用户上传了一段8秒的录音——那是他已故母亲生前在家庭聚会中的一句玩笑话。他输入了这样一段文字&#xff1a;“宝贝&#xff0c;生日快乐&#xff0c;妈妈永远爱你。”点击生成后&#xff0c;熟悉的音色…

作者头像 李华
网站建设 2026/4/22 9:55:36

49、基于 Pthreads 的多线程编程 II - 同步

基于 Pthreads 的多线程编程 II - 同步 在多线程编程中,同步是一个至关重要的话题。本文将深入探讨使用互斥锁(mutex)进行线程同步的相关内容,包括优先级反转问题、看门狗定时器、优先级继承以及互斥锁的其他变体。 1. 进程间通信与互斥锁 在进行进程间通信(IPC)时,建…

作者头像 李华
网站建设 2026/4/21 8:30:54

56、Linux 系统中的 CPU 调度与高级文件 I/O 技术

Linux 系统中的 CPU 调度与高级文件 I/O 技术 1. Linux 中的 CPU 调度 在 Linux 系统里,设置线程(或进程)的调度策略和优先级时,需要以 root 权限运行。现代为线程赋予特权的方式是通过 Linux 能力模型(Linux Capabilities model),具备 CAP_SYS_NICE 能力的线程可以…

作者头像 李华
网站建设 2026/4/18 8:23:32

59、高级文件 I/O 技术全解析

高级文件 I/O 技术全解析 在 Linux 系统编程中,高效的文件 I/O 操作至关重要。本文将深入介绍几种高级文件 I/O 技术,包括内存映射、直接 I/O(DIO)、异步 I/O(AIO)等,并对它们进行详细的比较和分析。 1. 内存映射(mmap) 内存映射(mmap)是一种将文件或设备映射到进…

作者头像 李华