RocketMQ的事务消息

1. 事务消息整体流程

事务消息是 RocketMQ 提供的分布式事务解决方案,用于保证生产者本地事务与消息发送的最终一致性。核心流程如下:

  1. 发送半消息:生产者先发送一条 "预备消息" 到 Broker,此时消息对消费者不可见。
  1. 执行本地事务:Broker 确认收到半消息后,生产者执行本地业务逻辑(如数据库操作)。
  1. 提交或回滚:根据本地事务的执行结果(成功 / 失败),生产者向 Broker 发送提交或回滚指令。
  1. 状态回查(关键机制):若生产者未及时返回状态(如服务宕机),Broker 会定时回查事务状态。

2. 核心方法解析

executeLocalTransaction方法

java

typescript 复制代码
@Override
public LocalTransactionState executeLocalTransaction(Message msg, Object arg) {
    try {
        // 1. 解析消息获取业务参数
        String projectId = parseProjectId(msg);
        String enterpriseId = parseEnterpriseId(msg);
        
        // 2. 执行本地数据库操作(带事务)
        // 例如:更新投标状态为"处理中"
        boolean success = bidService.updateBidStatus(projectId, enterpriseId, "PROCESSING");
        
        // 3. 根据本地事务结果返回状态
        return success ? LocalTransactionState.COMMIT_MESSAGE 
                       : LocalTransactionState.ROLLBACK_MESSAGE;
    } catch (Exception e) {
        // 异常时返回 UNKNOW,触发后续回查
        return LocalTransactionState.UNKNOW;
    }
}

Java

  • 作用:在发送半消息成功后,执行本地事务并返回状态。
  • 返回值
  • COMMIT_MESSAGE:提交事务,消息对消费者可见。
  • ROLLBACK_MESSAGE:回滚事务,消息被丢弃。
  • UNKNOW:状态未知,触发 Broker 的回查机制。

checkLocalTransaction方法

java

typescript 复制代码
@Override
public LocalTransactionState checkLocalTransaction(MessageExt msg) {
    // 查询数据库确认投标状态
    boolean isCommitted = getBidStatusFromDB(msg);
    return isCommitted ? LocalTransactionState.COMMIT_MESSAGE 
                       : LocalTransactionState.ROLLBACK_MESSAGE;
}


// 示例:从数据库查询投标状态
private boolean getBidStatusFromDB(MessageExt msg) {
    String projectId = parseProjectId(msg);
    String enterpriseId = parseEnterpriseId(msg);
    // 查询数据库,例如:
    // Bid bid = bidRepository.findByProjectIdAndEnterpriseId(projectId, enterpriseId);
    // return bid.getStatus().equals("SUCCESS");
    // 实际实现需根据业务表结构调整
}

Java

  • 作用:处理 Broker 的状态回查请求,通过查询数据库确认本地事务的最终状态。
  • 触发条件:当 Broker 长时间未收到生产者的提交 / 回滚指令(如生产者服务宕机),会按配置的间隔(默认 60s)调用此方法。

3. 事务回查机制的关键点

  1. 服务宕机场景
  • 若生产者在执行本地事务后、返回状态前宕机,Broker 无法感知事务结果。
  • 此时 Broker 会定时回查,通过checkLocalTransaction读取数据库真实状态。
  1. 幂等性保障
  • 回查可能被多次调用(如网络波动),因此checkLocalTransaction必须是幂等的。
  • 通常通过数据库唯一索引或状态字段(如status)保证重复查询结果一致。
  1. 数据库一致性
  • 本地事务需保证数据库操作的原子性(如使用数据库事务)。
  • 示例中,若更新投标状态失败,应返回ROLLBACK_MESSAGE,避免消息被消费。

4. 实际应用建议

  1. 状态设计
  • 在业务表中增加状态字段(如bid_status),记录事务处理阶段(如INITPROCESSINGSUCCESSFAILED)。
  1. 日志记录
  • 在关键步骤添加日志(如事务执行前后),便于问题排查。
  1. 回查配置优化
  • 通过TransactionMQProducer.setCheckThreadPool调整回查线程池大小。
  • 通过TransactionMQProducer.setCheckRequestHoldMax调整回查请求的最大等待时间。

总结

事务回查机制是 RocketMQ 事务消息的核心保障,通过定时检查数据库状态,解决了服务故障导致的状态丢失问题,确保分布式事务的最终一致性。

相关推荐
Victor3563 分钟前
MySQL(187)如何使用pt-query-digest进行查询分析?
后端
Victor3566 分钟前
MySQL(188)如何使用MySQL的慢查询工具?
后端
CF14年老兵25 分钟前
5 个最佳工具,可立即从代码生成 API 文档
前端·后端·api
麦兜*7 小时前
Spring Boot整合PyTorch Pruning工具链,模型瘦身手术
java·pytorch·spring boot·后端·spring cloud·ai编程·剪枝
杨荧9 小时前
基于大数据的美食视频播放数据可视化系统 Python+Django+Vue.js
大数据·前端·javascript·vue.js·spring boot·后端·python
Re27510 小时前
我用4碗面讲清HTTP的四大请求方法:GET/POST/PUT/DELETE
后端
有梦想的攻城狮10 小时前
spring中的ApplicationRunner接口详解
java·后端·spring·runner·application
程序视点10 小时前
设计模式之原型模式!附Java代码示例!
java·后端·设计模式
用户214118326360211 小时前
AI 驱动开发:20 分钟搞定智能发票申请单系统
后端
G探险者11 小时前
Java 中 null 值在 JSON 输出时丢失的坑:一次 Object 参数 + Fastjson 多态的血泪教训
后端