【RocketMQ 生产者和消费者】- 事务消息的使用


本文章基于 RocketMQ 4.9.3

1. 前言

上两篇文章我们分析了延时消息的基本使用和原理,这篇文章就来看下事务消息的基本使用。

2. 事务消息

首先介绍下什么是事务消息,也可以看官网的介绍:事务消息发送

事务消息可以用在消息队列中确保消息传递和业务操作原子性、一致性,尤其适用于分布式系统环境,特别是消息上下游服务相互依赖的场景。比如下面的两个场景:

  • 订单支付: 比如上游支付订单之后扣减本地库存,更新订单状态,接着通知下游服务比如物流系统做后续的处理,这种情况下可以先发送一条事务消息,然后执行本地事务逻辑 (扣减库存,更新订单状态),做完之后根据本地事务消息的结果决定这条消息能不能被下游的服务看到,如果本地事务执行失败,那么下游也不需要继续处理。
  • 金融系统中的转账业务:转账同时扣减转出账户余额和增加转入账户余额。用事务消息在发送转账成功消息之前执行本地账户余额更新事务,确保消息发送与账户余额变更操作的一致性。

上面说的本地事务可以是操作数据库、操作缓存等等,反正就是一个方法,根据业务来处理,下面就是大致的流程,这个图也是直接用官方的图了。

上面涉及到几个名词下面解释下:

  • 半事务消息:半事务消息是指暂不能投递的消息,当生产者发送半事务消息到 Broker后,Broker 收到消息会将其先存储起来,但不会马上将该消息投递给消费者,此时消费者对这条消息不可见。只有生产者对该消息进行二次确认(即提交或回滚操作)后,Broker 才会根据确认结果进行相应处理。
  • 本地事务:本地事务就是本地的逻辑,当生产者执行完本地事务之后,根据执行结果返回对应的状态码,broker 会根据对应结果来处理这条半事务消息。本地事务有三种状态:
    • UNKNOW 表示中间状态,代表需要通过事务回查来确定最终的执行结果;
    • COMMIT_MESSAGE 表示本地事务执行成功,此消息可以被消费者消费;
    • ROLLBACK_MESSAGE 表示事务回滚,这条消息会被删掉,不会被消费者消费。
  • 事务回查:如果由于网络闪断、生产者应用重启等原因,导致某条事务消息的二次确认丢失,Broker 端会通过扫描发现某条消息长期处于 "半事务消息"时,需要主动向消息生产者询问该消息的最终状态(Commit 或是 Rollback),如果业务阻塞也可以通过这种方式给一个兜底的结果,在 broker.conf 可以配置 transactionCheckInterval 回查的时间间隔,根据自己实际业务决定。

下面把整个流程串起来:

  1. 生产者将半事务消息发送至 broker。
  2. broker 将这条消息持久化,这条消息属于半事务消息,还不可被消费者消费。
  3. 生产者通过 executeLocalTransaction 开始执行本地事务逻辑。
  4. 本地事务执行完成后,生产者向 broker 提交二次确认结果(Commit 或是 Rollback),服务端走不同的处理逻辑:
    • 二次确认结果为 COMMIT_MESSAGE:服务端将半事务消息标记为可投递,并投递给消费者。
    • 二次确认结果为 ROLLBACK_MESSAGE:服务端将回滚事务,不会将半事务消息投递给消费者。
    • 二次确认结果为 UNKNOW:服务端将回滚事务,不会将半事务消息投递给消费者。
  5. 在断网或者是生产者应用重启的特殊情况下,broker 迟迟收不到事务的二次提交结果,又或者如果事务提交结果给的是 UNKNOW,那么服务端会对生产者发起事务回查。
  6. 为了避免单个消息被检查太多次而导致半队列消息累积,默认将单个消息的检查次数限制为 15 次,但是用户可以通过 Broker 配置文件的 transactionCheckMax 参数来修改检查次数。如果回查次数到达上限了,那么 broker 就丢掉这条消息,然后在默认情况下同时打印错误日志。不过我们可以通过重写 AbstractTransactionalMessageCheckListener、 类来修改回查上限之后要做什么。
  7. 生产者收到消息回查后,通过 checkLocalTransaction 检查对应消息的本地事务执行的最终结果。
  8. 根据检查出来的结果进行二次提交,这时候又回到了 4 的逻辑。

3. 示例

下面看下事务消息的使用,首先在 broker 配置下消息回查的时间。

java 复制代码
transactionCheckInterval=20000

然后设置下生产者,生产者一共发送 10 条消息,下标从 0 开始计算,然后每一条消息发送的 tag 不一样。

java 复制代码
public class TransactionProducer {
    public static void main(String[] args) throws MQClientException, InterruptedException {
        TransactionListener transactionListener = new TransactionListenerImpl();
        // 事务消息生产者
        TransactionMQProducer producer = new TransactionMQProducer("transactionMQProducer");
        // 执行本地回查的线程池
        ExecutorService executorService = new ThreadPoolExecutor(2, 5, 100, TimeUnit.SECONDS, new ArrayBlockingQueue<Runnable>(2000), new ThreadFactory() {
            @Override
            public Thread newThread(Runnable r) {
                Thread thread = new Thread(r);
                thread.setName("client-transaction-msg-check-thread");
                return thread;
            }
        });
        producer.setNamesrvAddr("localhost:9876");
        producer.setExecutorService(executorService);
        // 设置事务监听器
        producer.setTransactionListener(transactionListener);
        // 设置本地事务回查时间间隔
        producer.start();

        String[] tags = new String[] {"TagA", "TagB", "TagC", "TagD", "TagE"};
        for (int i = 0; i < 10; i++) {
            try {
                // 消息发送
                Message msg =
                    new Message("TopicTest1234", tags[i % tags.length], "KEY" + i,
                        ("Hello RocketMQ " + i).getBytes(RemotingHelper.DEFAULT_CHARSET));
                SendResult sendResult = producer.sendMessageInTransaction(msg, null);
                System.out.printf(now() + " %s%n", sendResult);

                Thread.sleep(10);
            } catch (MQClientException | UnsupportedEncodingException e) {
                e.printStackTrace();
            }
        }

        for (int i = 0; i < 100000; i++) {
            Thread.sleep(1000);
        }
        producer.shutdown();
    }
}

接下来初始化事务监听器 TransactionListenerImpl,实现 TransactionListener 接口,在里面定义 executeLocalTransaction 本地事务执行的逻辑以及 checkLocalTransaction 本地事务回查的逻辑。

java 复制代码
public class TransactionListenerImpl implements TransactionListener {
    private AtomicInteger transactionIndex = new AtomicInteger(0);

    private ConcurrentHashMap<String, Integer> localTrans = new ConcurrentHashMap<>();

    /**
     * 执行本地事务
     * @param msg Half(prepare) message
     * @param arg Custom business parameter
     * @return
     */
    @Override
    public LocalTransactionState executeLocalTransaction(Message msg, Object arg) {
        // 执行本地事务, 根据不同下标在下面的 checkLocalTransaction 方法走不同的逻辑
        int value = transactionIndex.getAndIncrement();
        int status = value % 3;
        localTrans.put(msg.getTransactionId(), status);
        // 全部返回 UNKNOW, 意味者要通过事务回查 checkLocalTransaction 方法去检查本地事务的提交结果
        return LocalTransactionState.UNKNOW;
    }

    /**
     * 本地事务回查
     * @param msg Check message
     * @return
     */
    @Override
    public LocalTransactionState checkLocalTransaction(MessageExt msg) {
        System.out.println(now() + " checkLocalTransaction 执行本地事务回查, 当前消息事务 ID: " + msg.getTransactionId());
        Integer status = localTrans.get(msg.getTransactionId());
        if (null != status) {
            switch (status) {
                case 0:
                    return LocalTransactionState.UNKNOW;
                case 1:
                    return LocalTransactionState.COMMIT_MESSAGE;
                case 2:
                    return LocalTransactionState.ROLLBACK_MESSAGE;
                default:
                    return LocalTransactionState.COMMIT_MESSAGE;
            }
        }
        return LocalTransactionState.COMMIT_MESSAGE;
    }
}

executeLocalTransaction 执行本地事务的时候,会先通过 transactionIndex + 1 算出这条消息是哪个下标,然后在事务回查 checkLocalTransaction 的时候将所有消息分为三类来处理。

  • 0:LocalTransactionState.UNKNOW
  • 1:LocalTransactionState.COMMIT_MESSAGE
  • 2:LocalTransactionState.ROLLBACK_MESSAGE

大致流程就是执行本地事务的时候 LocalTransactionState.UNKNOW,broker 会对每一条消息都进行回查,然后在回查里面会根据不同的下标来返回不同的结果,结合 executeLocalTransaction 方法,可以大概看出消息 0、3、6、9 会不断回查,1、4、7 回查之后发现是 COMMIT_MESSAGE,消息提交,消费者可以消费到这几条消息,2、5、8 回查之后是 ROLLBACK_MESSAGE,这几条消息会被丢掉,后续回查也不会对这几条消息发起回查了。

然后是消费者,消费者比较简单,就是订阅 TopicTest1234 下面的所有 tags 进行消费。

java 复制代码
public class TransactionConsumer {

    public static void main(String[] args) throws InterruptedException, MQClientException {

        DefaultMQPushConsumer consumer = new DefaultMQPushConsumer("testGroupConsumer");
        consumer.setNamesrvAddr("localhost:9876");
        consumer.setMessageModel(MessageModel.CLUSTERING);
        consumer.subscribe("TopicTest1234", "*");
        consumer.registerMessageListener(new MessageListenerConcurrently() {

            @Override
            public ConsumeConcurrentlyStatus consumeMessage(List<MessageExt> msgs,
                ConsumeConcurrentlyContext context) {
                for (int i = 0; i < msgs.size(); i++) {
                    MessageExt msg = msgs.get(i);
                    System.out.printf("%s, %s Receive New Messages: %s %n", now(),  Thread.currentThread().getName(), new String(msg.getBody(), StandardCharsets.UTF_8));
                    context.setAckIndex(i);
                }

                return ConsumeConcurrentlyStatus.CONSUME_SUCCESS;
            }
        });

        consumer.start();

        System.out.printf("Consumer Started.%n");
    }

    public static String now(){
        return new SimpleDateFormat("yyyy-MM-dd HH:mm:ss:SSS").format(new Date());
    }
}

输出结果如下,首先是消费者,可以看到确实只消费了 1、4、7 三条消息。

然后是生产者,生产者发送消息之后由于本地事务执行都是返回的 `LocalTransactionState.UNKNOW``,所以经过一定时间后会消息回查,这里为什么不是 20s,后面源码会分析。

经过事务回查之后,发现 2、5、8 返回了 ROLLBACK_MESSAGE,于是后续不会再回查这几条消息,而 0、3、6、9 由于还是返回 UNKNOW,于是就一直不断回查,但是回查次数有上限,所以也不会一直回查,不过时间太长了,就不贴出所有输出。

4. 使用限制

这里的限制也是代码给的文档里面标注出来的,下面贴出来:

  • 事务消息不支持 延时消息批量消息
  • 提交给用户的目标主题消息可能会失败,它的高可用性通过 RocketMQ 本身的高可用性机制来保证,如果希望确保事务消息不丢失、并且事务完整性得到保证,建议使用 同步的双重写入机制
  • 事务消息的生产者 ID 不能与其他类型消息的生产者 ID 共享。与其他类型的消息不同,事务消息允许 反向查询、MQ 服务器能通过它们的生产者 ID 查询到消费者。

5. 小结

好了,到这里事务消息的使用和大致流程已经介绍完,下一篇文章就要进入源码分析环节。

如有错误,欢迎指出!!!

相关推荐
Via_Neo1 小时前
接雨水问题 + 输入优化
java·开发语言·算法
xufengzhu1 小时前
多层Module依赖项目Maven编译错误的解决方案
java·maven
吃鱼不吐刺.1 小时前
阻塞队列。
java·开发语言
啦啦啦_99991 小时前
3. AI面试题之 FunctionCall
java
半夜修仙1 小时前
总结一下 Spring 中存取 Bean 的相关注解, 以及这些注解的用法.
java·笔记·学习·spring
彭于晏Yan1 小时前
Spring Cloud Security:Oauth2令牌存储
java·spring boot·spring cloud
不光头强1 小时前
ArrayList知识点
java·开发语言·windows
皙然2 小时前
吃透 Java 泛型
java
斌糖雪梨2 小时前
invokeBeanFactoryPostProcessors(beanFactory); 方法详解
java·后端·spring