如何实现顺序消息? 需要程序保证发送和消费的是同一个 Queue
rocketmq默认发送的消息是进入多个消息队列,然后消费端多线程并发消费,所以默认情况,不是順序消费消息的;有時候,我们需要顺序消费一批消息,比如电商系统 订单创建、支付、完成操作,需要順序执行;
RocketMQTemplate给我们提供了SendOrderly方法(有多個重载),来实现发送顺序消息;包括以下:
syncSendOrderly,发送同步顺序消息;
asyncSendOrderly,发送异步顺序消息;
sendOneWayOrderly,发送单向顺序消息;
一般我们用syncSendOrderly方法发送同步顺序消息。
参数一:topic 如果想添加tag,可以使用"topic:tag"的写法
参数二:消息内容
参数三:hashKey 使用此参数选择队列。 例如:orderId,productId…
因为broker会管理多个消息队列,这个hashKey参数,主要用来计算选择队列
的,一般可以把订单ID,产品ID作为参数值;发送到一个队列,这样方便搞顺序队列;以及消费端接收的时候,默认是并发多线程
去接收消息。
RocketMQMessageListener有个属性consumeMode
,默认是ConsumeMode.CONCURRENTLY ,我们要改成ConsumeMode.ORDERLY,单线程顺序接收消息;
下面来介绍下 springboot+rockermq 整合实现 顺序消息的发送与消费
一、创建Springboot项目,添加rockermq 依赖
<!--rocketMq依赖-->
<dependency>
<groupId>org.apache.rocketmq</groupId>
<artifactId>rocketmq-spring-boot-starter</artifactId>
<version>2.2.1</version>
</dependency>
二、配置rocketmq
# 端口
server:
port: 8083
# 配置 rocketmq
rocketmq:
name-server: 127.0.0.1:9876
#生产者
producer:
#生产者组名,规定在一个应用里面必须唯一
group: group1
#消息发送的超时时间 默认3000ms
send-message-timeout: 3000
#消息达到4096字节的时候,消息就会被压缩。默认 4096
compress-message-body-threshold: 4096
#最大的消息限制,默认为128K
max-message-size: 4194304
#同步消息发送失败重试次数
retry-times-when-send-failed: 3
#在内部发送失败时是否重试其他代理,这个参数在有多个broker时才生效
retry-next-server: true
#异步消息发送失败重试的次数
retry-times-when-send-async-failed: 3
三、新建一个 controller 来做消息发送
package com.example.springbootrocketdemo.controller;
import org.apache.rocketmq.spring.core.RocketMQTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
/**
* 模拟两个订单发送消息
*
* 顺序信息的三种方式:同步、异步、单向
* syncSendOrderly,发送同步顺序消息;
* asyncSendOrderly,发送异步顺序消息;
* sendOneWayOrderly,发送单向顺序消息;
* 一般我们用第一种发送同步顺序消息;
* @author qzz
*/
@RestController
public class RocketMQOrderCOntroller {
@Autowired
private RocketMQTemplate rocketMQTemplate;
/**
* 发送同步顺序消息
*/
@RequestMapping("/testSyncOrderSend")
public void testSyncSend(){
//参数一:topic 如果想添加tag,可以使用"topic:tag"的写法
//参数二:消息内容
//参数三:hashKey 用来计算决定消息发送到哪个消息队列, 一般是订单ID,产品ID等
rocketMQTemplate.syncSendOrderly("test-topic-orderly","111111创建","111111");
rocketMQTemplate.syncSendOrderly("test-topic-orderly","111111支付","111111");
rocketMQTemplate.syncSendOrderly("test-topic-orderly","111111完成","111111");
rocketMQTemplate.syncSendOrderly("test-topic-orderly","222222创建","222222");
rocketMQTemplate.syncSendOrderly("test-topic-orderly","222222支付","222222");
rocketMQTemplate.syncSendOrderly("test-topic-orderly","222222完成","222222");
}
}
四、创建消费端,监听消息,消费消息
package com.example.springbootrocketdemo.config;
import org.apache.rocketmq.spring.annotation.ConsumeMode;
import org.apache.rocketmq.spring.annotation.RocketMQMessageListener;
import org.apache.rocketmq.spring.core.RocketMQListener;
import org.springframework.stereotype.Service;
/**
* 消费顺序消息
* 配置RocketMQ监听
*
* ConsumeMode.ORDERLY:顺序消费
* @author qzz
*/
@Service
@RocketMQMessageListener(consumerGroup = "test",topic = "test-topic-orderly",consumeMode = ConsumeMode.ORDERLY)
public class RocketMQCommonConsumerListener implements RocketMQListener<String> {
@Override
public void onMessage(String s) {
System.out.println("consumer 顺序消费,收到消息:"+s);
}
}
五、启动服务,测试 顺序消息 发送与消费
测试成功!