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
相关推荐
世间万物皆对象3 小时前
Spring Boot核心概念:日志管理
java·spring boot·单元测试
懒洋洋大魔王4 小时前
RocketMQ的使⽤
java·rocketmq·java-rocketmq
qq_17448285754 小时前
springboot基于微信小程序的旧衣回收系统的设计与实现
spring boot·后端·微信小程序
代码小鑫6 小时前
A043-基于Spring Boot的秒杀系统设计与实现
java·开发语言·数据库·spring boot·后端·spring·毕业设计
真心喜欢你吖6 小时前
SpringBoot与MongoDB深度整合及应用案例
java·spring boot·后端·mongodb·spring
周全全7 小时前
Spring Boot + Vue 基于 RSA 的用户身份认证加密机制实现
java·vue.js·spring boot·安全·php
飞升不如收破烂~8 小时前
Spring boot常用注解和作用
java·spring boot·后端
计算机毕设源码qq-38365310418 小时前
(附项目源码)Java开发语言,215 springboot 大学生爱心互助代购网站,计算机毕设程序开发+文案(LW+PPT)
java·开发语言·spring boot·mysql·课程设计
岁岁岁平安8 小时前
springboot实战(15)(注解@JsonFormat(pattern=“?“)、@JsonIgnore)
java·spring boot·后端·idea
潜洋12 小时前
Spring Boot教程之五:在 IntelliJ IDEA 中运行第一个 Spring Boot 应用程序
java·spring boot·后端