RocketMQ事务消息

恕我直言,网上搜到的关于rocketMQ事务消息的实例都是屁话(也有可能是我菜),毫无实战可言

这里不做事务消息的原理和用途介绍(原理网上一大堆),这里解决 RocketMQLocalTransactionListener 一个 project 里只能有一个的问题。

以下内容全部以 springboot-rocketmq 依赖为基础

xml 复制代码
<dependency>
    <groupId>org.apache.rocketmq</groupId>
    <artifactId>rocketmq-spring-boot-starter</artifactId>
</dependency>

好像因为 TransactionMQProducer 在执行事务时不能被共享, 一个项目里如果包含多个 @RocketMQTransactionListener, 启动将会报错。

那我们的项目里如果不止一个业务场景需要发送事务消息,那怎么办呢?

1.仿照 RocketMQLocalTransactionListener 自定义一个相同的接口 TransactionMessageCommonListener

java 复制代码
public interface TransactionMessageCommonListener<T> {

    RocketMQLocalTransactionState executeLocalTransaction(MessageHeaders headers, T message, Object args);

    RocketMQLocalTransactionState checkLocalTransaction(MessageHeaders headers, T message);

}

2.定义一个实现了 RocketMQLocalTransactionListener 接口的默认实现类

java 复制代码
@Component
@RocketMQTransactionListener
public class DefaultRocketMQTransactionMessageImpl implements RocketMQLocalTransactionListener {

    @Override
    public RocketMQLocalTransactionState executeLocalTransaction(Message msg, Object arg) {
        String listenerName = (String) msg.getHeaders().get("listener");
        if (null == listenerName) {
            throw new RuntimeException("not params transactionMessageListener");
        }
        RocketMQLocalTransactionState state;
        try {
            TransactionBody transactionBody = transactionBodyMap.get(listenerName);
            if (null == transactionBody) {
                throw new RuntimeException("not match condition transactionMessageListener");
            }
            state = transactionBody.getListener().executeLocalTransaction(msg.getHeaders(), MessageUtil.doConvertMessage(transactionBody.getType(), msg), arg);
        } catch (Exception e) {
            log.error("rocket transaction message executeLocal error:{}", e.getMessage());
            return RocketMQLocalTransactionState.ROLLBACK;
        }
        return state;
    }

    @Override
    public RocketMQLocalTransactionState checkLocalTransaction(Message msg) {
        String listenerName = (String) msg.getHeaders().get("listener");
        if (null == listenerName) {
            throw new RuntimeException("not params transactionMessageListener");
        }
        RocketMQLocalTransactionState state;
        try {
            TransactionBody transactionBody = transactionBodyMap.get(listenerName);
            if (null == transactionBody) {
                throw new RuntimeException("not match condition transactionMessageListener");
            }
            state = transactionBody.getListener().checkLocalTransaction(msg.getHeaders(), MessageUtil.doConvertMessage(transactionBody.getType(), msg));
        } catch (Exception e) {
            log.error("rocket transaction message checkLocal error:", e);
            return RocketMQLocalTransactionState.ROLLBACK;
        }
        return state;
    }
}

3.定义一个包装类 TransactionBody,方便后续处理。再在上一步的默认实现类里对我们自定义的 TransactionMessageCommonListener 接口做一个所有实现类的缓存

java 复制代码
public class TransactionBody {

    private TransactionMessageCommonListener listener;

    private Type type;

    public TransactionMessageCommonListener getListener() {
        return listener;
    }

    public void setListener(TransactionMessageCommonListener listener) {
        this.listener = listener;
    }

    public Type getType() {
        return type;
    }

    public void setType(Type type) {
        this.type = type;
    }

    public TransactionBody (){};

    public TransactionBody(TransactionMessageCommonListener listener) {
        this.listener = listener;
        this.type = ((ParameterizedType) listener.getClass().getGenericInterfaces()[0]).getActualTypeArguments()[0];
    }
}
java 复制代码
private Map<String, TransactionBody> transactionBodyMap;

//该方法是在spring启动时完成调用的
//如果你的 TransactionMessageCommonListener 实现类会加入spring bean管理,TransactionBody::new 可以替换成 SpringBeanUtil.getBean(item.getClass)
@Autowired(required = false)
public void setTransactionMessageListeners(List<TransactionMessageCommonListener<?>> transactionMessageListeners) {
	if (!transactionMessageListeners.isEmpty()) {
		transactionBodyMap = transactionMessageListeners.stream()
				.collect(Collectors.toMap(item -> item.getClass().getSimpleName(), TransactionBody::new));
	}
}

4.定义发送消息的方法

java 复制代码
//msgBody是我自己封装的消息包装类,你可以根据你自己需要封装一个或者不封装
//msgBody.getContentJson()是你发送的业务数据,你可以换成你自己的
@Override
public <T> void sendTransaction(MsgBody msgBody, Object args, Class<? extends TransactionMessageCommonListener<T>> transactionMessageListener) {
    rocketMQTemplate.sendMessageInTransaction(msgBody.getTopic() + ":" + msgBody.getTags(), MessageBuilder.withPayload(msgBody.getContentJson()).setHeader("listener", transactionMessageListener.getSimpleName()).build(), args);
}

到这里对事务消息的改造就完成了,接下来是使用

1.发送一个事务消息,并指定本地事务提交的实现类是哪一个

java 复制代码
Order order = new Order("A123", "测试订单", LocalDateTime.now());
rocketService.sendTransaction(
        new MsgBody()
                .setContent(JSONObject.toJsonString(order))
                .setTopic("TEST_TX_TOPIC")
                .setTags("TEST_TX_TAG"),
        order,
        CreateOrderTransactionMessageListener.class
);

2.定义 CreateOrderTransactionMessageListener 实现刚才我们定义好的 TransactionMessageCommonListener 接口

java 复制代码
public class CreateOrderTransactionMessageListener implements TransactionMessageCommonListener<Order> {
    @Override
    public RocketMQLocalTransactionState executeLocalTransaction(MessageHeaders headers, Order message, Object args) {
        
        //执行业务代码(paySuccessCrateOrder 方法包含了 @Transactional)
        boolean isSuccess = orderService.paySuccessCrateOrder((Order) args);
        
        return isSuccess ?  RocketMQLocalTransactionState.COMMIT : RocketMQLocalTransactionState.UNKNOWN;
        
    }

    @Override
    public RocketMQLocalTransactionState checkLocalTransaction(MessageHeaders headers, Order message) {
        
        //查询刚才的数据是否成功落地,成功就提交消息,没有就回滚消息
        Order order = orderService.queryOrderIsExist(message.getOrderId());
        
        return null != order ? RocketMQLocalTransactionState.COMMIT : RocketMQLocalTransactionState.ROLLBACK;
        
    }
}

以后项目里有其他业务场景需要发送事务消息,只需要实现一个 TransactionMessageCommonListener 接口,并在发送消息时指定这个实现类就好了。

该代码是一两年前写的,有一些地方可能设计并不合理或需要优化,请根据自己的实际需要进行修改。copy代码只能保证正常运行。

相关推荐
man20171 小时前
【2024最新】基于springboot+vue的闲一品交易平台lw+ppt
vue.js·spring boot·后端
hlsd#1 小时前
关于 SpringBoot 时间处理的总结
java·spring boot·后端
路在脚下@1 小时前
Spring Boot 的核心原理和工作机制
java·spring boot·后端
幸运小圣2 小时前
Vue3 -- 项目配置之stylelint【企业级项目配置保姆级教程3】
开发语言·后端·rust
前端SkyRain3 小时前
后端Node学习项目-用户管理-增删改查
后端·学习·node.js
提笔惊蚂蚁3 小时前
结构化(经典)软件开发方法: 需求分析阶段+设计阶段
后端·学习·需求分析
老猿讲编程3 小时前
Rust编写的贪吃蛇小游戏源代码解读
开发语言·后端·rust
黄小耶@3 小时前
python如何使用Rabbitmq
分布式·后端·python·rabbitmq
宅小海4 小时前
Scala-List列表
开发语言·后端·scala
蔚一5 小时前
Javaweb—Ajax与jQuery请求
前端·javascript·后端·ajax·jquery