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 事务消息的核心保障,通过定时检查数据库状态,解决了服务故障导致的状态丢失问题,确保分布式事务的最终一致性。

相关推荐
JuiceFS14 分钟前
合合信息:基于 JuiceFS 构建统一存储,支撑 PB 级 AI 训练
运维·后端
调试人生的显微镜20 分钟前
iOS 文件深度调试实战 查看用户文件 App 沙盒 系统文件与日志全指南
后端
aiopencode22 分钟前
没有 Mac,如何上架 iOS App?跨平台团队的全流程实践指南
后端
天天摸鱼的java工程师24 分钟前
如何防止重复提交订单?
java·后端·面试
星星电灯猴27 分钟前
Charles 抓不到包怎么办?详解原因与多维解决方案
后端
慕尘_30 分钟前
对于未来技术的猜想:Manus as a Service
前端·后端
秋水丶秋水37 分钟前
Python常见异常和处理方案
后端·python
程序无bug38 分钟前
pring Boot监控方案
java·后端
荔枝爱编程39 分钟前
高性能企业级消息中心架构实现与分享(二)
后端·消息队列·rocketmq
莹莹啦40 分钟前
Java 21 核心特性全景解析:LTS 版本的革命性升级
后端