RocketMQ整合SpringBoot事务消息

RocketMQ事务消息流程

发送半消息:生产者首先发送一个半消息(即预消息)到 Broker。半消息的作用是先占据一个位置,但不会被消费者消费,直到事务提交成功。

执行本地事务 :Broker 接收到半消息并确认后,回调生产者的 executeLocalTransaction 方法执行本地事务。

提交事务状态:根据本地事务的执行结果,生产者向 Broker 提交事务状态(COMMIT、ROLLBACK 或 UNKNOWN)。如果提交成功,半消息会被转换为正式消息,并且可以被消费者消费;如果回滚,半消息会被删除。

事务状态回查 :如果 Broker 未能及时收到生产者的提交状态,或者收到的是 UNKNOWN 状态,Broker 会定期回调生产者的 checkLocalTransaction 方法以确定事务的最终状态。

消费者消费消息:当事务消息被提交后,消费者可以正常消费这条消息。消费者接收并处理消息的逻辑与普通消息相同。

package com.example.springbootrocketmq.controller;

import com.example.springbootrocketmq.producer.RocketMQTransactionProducerService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

/**
 * 控制器类,用于触发事务消息发送
 * @author hrui
 * @date 2024/8/4 2:40
 */
@RestController
public class TransactionController {

    @Autowired
    private RocketMQTransactionProducerService producerService;

    @GetMapping("/send")
    public String sendTransactionMessage() {
        producerService.sendMessageInTransaction("transaction-topic", "事务消息负载");
        return "事务消息已发送";
    }
}

package com.example.springbootrocketmq.producer;

import com.example.springbootrocketmq.pojo.User;
import org.apache.rocketmq.client.producer.SendResult;
import org.apache.rocketmq.spring.core.RocketMQTemplate;
import org.apache.rocketmq.spring.support.RocketMQHeaders;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.messaging.Message;
import org.springframework.messaging.support.MessageBuilder;
import org.springframework.stereotype.Service;

import java.util.UUID;

/**
 * 事务消息生产者服务类
 * @author hrui
 * @date 2024/8/4 2:40
 */
@Service
public class RocketMQTransactionProducerService {

    @Autowired
    private RocketMQTemplate rocketMQTemplate;

    /**
     * 发送事务消息
     * @param topic 消息主题
     * @param msgBody 消息内容
     */
    public void sendMessageInTransaction(String topic, String msgBody) {
        //构建消息,指定消息体和事务ID
        //Message<String> message = MessageBuilder.withPayload(msgBody).build();
        Message<String> message = MessageBuilder.withPayload(msgBody)
                .setHeader(RocketMQHeaders.TRANSACTION_ID, UUID.randomUUID().toString())//可以在MQ消息头里放 一个key  一个value
                .build();

        //发送事务消息,rocketMQTemplate会触发事务监听器的执行
        //1.发送半消息
        //第一个参数可以传topic 也可以传topic+tag  第三个参数是业务参数(本地回调时候可以访问到,会传过来)
        SendResult sendResult = rocketMQTemplate.sendMessageInTransaction(topic+":"+"tags", message, new User("hrui", 18, "杭州"));//第三个是业务参数  业务里要带过去什么就带什么
        System.out.println("发送结果: " + sendResult);
    }
}

package com.example.springbootrocketmq.listener;

import org.apache.rocketmq.spring.annotation.RocketMQTransactionListener;
import org.apache.rocketmq.spring.core.RocketMQLocalTransactionListener;
import org.apache.rocketmq.spring.core.RocketMQLocalTransactionState;
import org.apache.rocketmq.spring.support.RocketMQUtil;
import org.springframework.messaging.Message;
import org.springframework.messaging.converter.StringMessageConverter;
import org.springframework.stereotype.Component;

/**
 * 事务监听器类,用于管理本地事务状态
 * @author hrui
 * @date 2024/8/4 2:41
 */
@RocketMQTransactionListener(rocketMQTemplateBeanName = "rocketMQTemplate")
public class RocketMQTransactionListenerImpl implements RocketMQLocalTransactionListener {

    @Override
    public RocketMQLocalTransactionState executeLocalTransaction(Message message, Object o) {
        // 将Spring的message转为RocketMQ的Message
        org.apache.rocketmq.common.message.Message rocketMsg = RocketMQUtil.convertToRocketMessage(new StringMessageConverter(), "UTF-8", o.toString(), message);
        // 执行本地事务
        System.out.println("执行本地事务: " + new String(rocketMsg.getBody()));
        try {
            // 执行实际的本地事务逻辑
            boolean success = true; // 执行本地事务(例如数据库操作)
            if (success) {
                return RocketMQLocalTransactionState.COMMIT;
            } else {
                return RocketMQLocalTransactionState.ROLLBACK;
            }
        } catch (Exception e) {
            e.printStackTrace();
            return RocketMQLocalTransactionState.UNKNOWN;
        }
    }

    @Override
    public RocketMQLocalTransactionState checkLocalTransaction(Message message) {
        // 检查事务状态
        System.out.println("检查本地事务: " + new String(message.getPayload().toString()));
        try {
            // 检查本地事务执行结果
            boolean success = true; // 检查本地事务状态(例如查询数据库)
            if (success) {
                return RocketMQLocalTransactionState.COMMIT;
            } else {
                return RocketMQLocalTransactionState.ROLLBACK;
            }
        } catch (Exception e) {
            e.printStackTrace();
            return RocketMQLocalTransactionState.UNKNOWN;
        }
    }
}

消费者

package com.example.springbootrocketmq.consumer;

import org.apache.rocketmq.spring.annotation.RocketMQMessageListener;
import org.apache.rocketmq.spring.core.RocketMQListener;
import org.springframework.stereotype.Component;

/**
 * 消费者服务类,用于消费来自主题的消息
 * @author hrui
 * @date 2024/8/4 2:42
 */
@Component
@RocketMQMessageListener(topic = "transaction-topic", consumerGroup = "transaction-consumer-group")
public class RocketMQTransactionConsumerService implements RocketMQListener<String> {

    @Override
    public void onMessage(String message) {
        System.out.println("接收到事务消息: " + message);
    }
}

测试时候是在一个应用中做的测试,application.properties 加了ACL访问控制

复制代码
rocketmq.name-server=xxx.xxx.xxx:9876
rocketmq.producer.group=mq_producer_group_test
rocketmq.producer.access-key=xxxxx
rocketmq.producer.secret-key=xxxxx


rocketmq.consumer.access-key=xxxxx
rocketmq.consumer.secret-key=xxxxx
相关推荐
小蜗牛慢慢爬行几秒前
有关异步场景的 10 大 Spring Boot 面试问题
java·开发语言·网络·spring boot·后端·spring·面试
Allen Bright7 分钟前
Spring Boot 整合 RabbitMQ:手动 ACK 与 QoS 配置详解
spring boot·rabbitmq·java-rabbitmq
goTsHgo29 分钟前
在 Spring Boot 的 MVC 框架中 路径匹配的实现 详解
spring boot·后端·mvc
钱多多_qdd39 分钟前
spring cache源码解析(四)——从@EnableCaching开始来阅读源码
java·spring boot·spring
飞的肖1 小时前
前端使用 Element Plus架构vue3.0实现图片拖拉拽,后等比压缩,上传到Spring Boot后端
前端·spring boot·架构
Q_19284999061 小时前
基于Spring Boot的摄影器材租赁回收系统
java·spring boot·后端
gb42152872 小时前
springboot中Jackson库和jsonpath库的区别和联系。
java·spring boot·后端
黄油饼卷咖喱鸡就味增汤拌孜然羊肉炒饭11 小时前
SpringBoot如何实现缓存预热?
java·spring boot·spring·缓存·程序员
AskHarries13 小时前
Spring Cloud OpenFeign快速入门demo
spring boot·后端
isolusion14 小时前
Springboot的创建方式
java·spring boot·后端