文章目录
- 1 摘要
- 2 核心 Maven 依赖
- 3 核心代码
- 3.1 同步消息
- 3.2 异步消息
- 3.3 事务消息
- 4 测试
- 4.1 同步消息
- 4.2 异步消息
- 4.3 事务消息
- 5 注意事项
- 6 推荐参考资料
- 7 Github 源码
1 摘要
RocketMQ 支持多种推送消息的方式,可根据业务需要使用。本文将介绍SpringBoot 2 集成 RocketMQ 4.9 收发同步、异步以及事务消息
SpringBoot 2 集成 RocketMQ:
SpringBoot 2.x 简易集成 RocketMQ
2 核心 Maven 依赖
./demo-rocketmq-producer/pom.xml
./demo-rocketmq-consumer/pom.xml
<!-- rocketMQ -->
<dependency>
<groupId>org.apache.rocketmq</groupId>
<artifactId>rocketmq-spring-boot-starter</artifactId>
<version>${rocketmq.spring.version}</version>
</dependency>
其中 ${rocketmq.spring.version}
的版本为 2.2.1
,对应的 rocketmq-client 版本为 4.9.1
3 核心代码
3.1 同步消息
同步消息使用场景: 对业务数据具有高一致性要求
同步消息程序执行逻辑: 生产者先将消息发送到 RocketMQ (即 Broker) ,Broker 接收到之后再反馈生产者 ACK 码,生产者拿到反馈后,返回消息发送成功。生产者必须等待 Broker 的反馈结果才执行下一步。
发送端(Producer)
./demo-rocketmq-producer/src/main/java/com/ljq/demo/springboot/rocketmq/producer/common/rocketmq/RocketMQProducer.java
/**
* 发送同步消息
*
* @param message
* @return
*/
public SendResult sendSync(String message) {
log.info("RocketMQ producer, sync message, topic: {}, message: {}", RocketMQConst.TOPIC_SYNC, message);
return rocketMQTemplate.syncSend(RocketMQConst.TOPIC_SYNC, message);
}
调用方法
org.apache.rocketmq.spring.core.RocketMQTemplate#syncSend(java.lang.String, java.lang.Object)
接收端(Consumer)
./demo-rocketmq-consumer/src/main/java/com/ljq/springboot/rocketmq/consumer/common/rocketmq/RocketMQSyncConsumer.java
package com.ljq.springboot.rocketmq.consumer.common.rocketmq;
import com.ljq.springboot.rocketmq.consumer.common.constant.RocketMQConst;
import lombok.extern.slf4j.Slf4j;
import org.apache.rocketmq.spring.annotation.RocketMQMessageListener;
import org.apache.rocketmq.spring.core.RocketMQListener;
import org.springframework.stereotype.Component;
/**
* @Description: RocketMQ 同步消息消费者
* @Author: junqiang.lu
* @Date: 2021/12/2
*/
@Slf4j
@Component
@RocketMQMessageListener(topic = RocketMQConst.TOPIC_SYNC, consumerGroup = RocketMQConst.GROUP_CONSUMER_SYNC)
public class RocketMQSyncConsumer implements RocketMQListener<String> {
@Override
public void onMessage(String s) {
log.info("rocketMQ consumer, topic:{}, message:{}", RocketMQConst.TOPIC_SYNC, s);
}
}
3.2 异步消息
异步消息使用场景: 数据吞吐量大,保证数据最终一致性
异步消息程序执行逻辑: 生产者将消息发送给 Broker,无需等待 Broker 的反馈结果即可执行下一步程序。Broker 的反馈结果会通过异步的方式通知给生产者。
发送端(Producer)
./demo-rocketmq-producer/src/main/java/com/ljq/demo/springboot/rocketmq/producer/common/rocketmq/RocketMQProducer.java
/**
* 发送异步消息
*
* @param message
*/
public void sendAsync(String message) {
log.info("RocketMQ producer, async message, topic: {}, message: {}", RocketMQConst.TOPIC_ASYNC, message);
rocketMQTemplate.asyncSend(RocketMQConst.TOPIC_ASYNC, message, new SendCallback() {
@Override
public void onSuccess(SendResult sendResult) {
log.info("async message send success !!! \n topic: {}, message: {}, result: {}",
RocketMQConst.TOPIC_ASYNC, message, sendResult);
}
@Override
public void onException(Throwable throwable) {
log.info("async message send fail ... \n topic: {}, message: {}, error: {}", RocketMQConst.TOPIC_ASYNC,
message, throwable.getMessage());
}
});
}
调用方法
org.apache.rocketmq.spring.core.RocketMQTemplate#asyncSend(java.lang.String, java.lang.Object, org.apache.rocketmq.client.producer.SendCallback)
接收端(Consumer)
./demo-rocketmq-consumer/src/main/java/com/ljq/springboot/rocketmq/consumer/common/rocketmq/RocketMQAsyncConsumer.java
package com.ljq.springboot.rocketmq.consumer.common.rocketmq;
import com.ljq.springboot.rocketmq.consumer.common.constant.RocketMQConst;
import lombok.extern.slf4j.Slf4j;
import org.apache.rocketmq.spring.annotation.RocketMQMessageListener;
import org.apache.rocketmq.spring.core.RocketMQListener;
import org.springframework.stereotype.Component;
/**
* @Description: RocketMQ 异步消息消费者
* @Author: junqiang.lu
* @Date: 2021/12/2
*/
@Slf4j
@Component
@RocketMQMessageListener(topic = RocketMQConst.TOPIC_ASYNC, consumerGroup = RocketMQConst.GROUP_CONSUMER_ASYNC)
public class RocketMQAsyncConsumer implements RocketMQListener<String> {
@Override
public void onMessage(String s) {
log.info("rocketMQ consumer, topic:{}, message:{}", RocketMQConst.TOPIC_ASYNC, s);
}
}
3.3 事务消息
事务消息使用场景: 适用于需要用到事务的地方,支持对大型事务进行拆分,通过二阶段提交,实现事务最终一致性
事务消息程序执行逻辑: 生产者向 Broker 发送消息,此时消息暂存在 Broker,并未推送给消费者,待生产者端本地事务执行成功之后再向 Broker 发送提交指令,此时 Broker 才会将消息推送给消费者。
重试机制: 当生产者端的事务提交失败后,后间隔一定的时间(默认 60 秒),对事务状态进行检查,如果此时本地事务提交成功,则会向 Broker 发送提交指令;若本地事务仍然提交失败,则继续间隔一定时间后进行重试,直到达到最大重试次数(默认为 15 次)。
发送端(Producer)
./demo-rocketmq-producer/src/main/java/com/ljq/demo/springboot/rocketmq/producer/common/rocketmq/RocketMQProducer.java
/**
* 发送事务消息
*
* @param message
*/
public TransactionSendResult sendTransaction(String message) {
String transactionId = UUID.randomUUID().toString();
log.info("{RocketMQ producer, transaction message, topic: {}, message: {}, transactionId: {}}",
RocketMQConst.TOPIC_TRANSACTION, message, transactionId);
Message<String> msg = MessageBuilder.withPayload(message).setHeader(RocketMQHeaders.TRANSACTION_ID,
transactionId).build();
return rocketMQTemplate.sendMessageInTransaction(RocketMQConst.TOPIC_TRANSACTION, msg, message);
}
调用方法
org.apache.rocketmq.spring.core.RocketMQTemplate#sendMessageInTransaction
生产者端事务监听
./demo-rocketmq-producer/src/main/java/com/ljq/demo/springboot/rocketmq/producer/common/rocketmq/RocketMQTransactionListenerImpl.java
package com.ljq.demo.springboot.rocketmq.producer.common.rocketmq;
import com.ljq.demo.springboot.rocketmq.producer.common.constant.RocketMQConst;
import lombok.extern.slf4j.Slf4j;
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.RocketMQHeaders;
import org.springframework.messaging.Message;
import java.util.Objects;
import java.util.concurrent.ConcurrentHashMap;
/**
* @Description: RocketMQ 事务监听器
* @Author: junqiang.lu
* @Date: 2021/12/6
*/
@Slf4j
@RocketMQTransactionListener()
public class RocketMQTransactionListenerImpl implements RocketMQLocalTransactionListener {
private ConcurrentHashMap<String, String> localTransaction = new ConcurrentHashMap<>();
@Override
public RocketMQLocalTransactionState executeLocalTransaction(Message message, Object o) {
String transactionId = String.valueOf(message.getHeaders().get(RocketMQHeaders.TRANSACTION_ID));
log.info("RocketMQ local transaction execute, message:{},transactionId: {}, param:{}", message.getPayload(),
transactionId, o);
// TODO 处理本地事务
String transactionParam = String.valueOf(o);
log.info("transactionParam:{}", transactionParam);
if (transactionParam.contains(RocketMQConst.TRANSACTION_STATUS_SUCCESS)) {
localTransaction.put(transactionId, RocketMQConst.TRANSACTION_STATUS_SUCCESS);
log.info("local transaction handle result: {}",RocketMQConst.TRANSACTION_STATUS_SUCCESS);
return RocketMQLocalTransactionState.COMMIT;
}
if (transactionParam.contains(RocketMQConst.TRANSACTION_STATUS_FAIL)) {
localTransaction.put(transactionId, RocketMQConst.TRANSACTION_STATUS_FAIL);
log.info("local transaction handle result: {}",RocketMQConst.TRANSACTION_STATUS_FAIL);
return RocketMQLocalTransactionState.ROLLBACK;
}
log.info("transaction status unknown");
return RocketMQLocalTransactionState.UNKNOWN;
}
@Override
public RocketMQLocalTransactionState checkLocalTransaction(Message message) {
String transactionId = String.valueOf(message.getHeaders().get(RocketMQHeaders.TRANSACTION_ID));
log.info("RocketMQ local transaction check, message:{},transactionId: {}", message.getPayload(), transactionId);
// TODO 校验本地事务
String transactionResult = localTransaction.getOrDefault(transactionId, "");
if (Objects.equals(transactionResult, RocketMQConst.TRANSACTION_STATUS_SUCCESS)) {
return RocketMQLocalTransactionState.COMMIT;
}
if (Objects.equals(transactionResult, RocketMQConst.TRANSACTION_STATUS_FAIL)) {
return RocketMQLocalTransactionState.ROLLBACK;
}
return RocketMQLocalTransactionState.UNKNOWN;
}
}
接收端(Consumer)
./demo-rocketmq-consumer/src/main/java/com/ljq/springboot/rocketmq/consumer/common/rocketmq/RocketMQTransactionConsumer.java
package com.ljq.springboot.rocketmq.consumer.common.rocketmq;
import com.ljq.springboot.rocketmq.consumer.common.constant.RocketMQConst;
import lombok.extern.slf4j.Slf4j;
import org.apache.rocketmq.spring.annotation.RocketMQMessageListener;
import org.apache.rocketmq.spring.core.RocketMQListener;
import org.springframework.stereotype.Component;
/**
* @Description: RocketMQ 事务消息消费者
* @Author: junqiang.lu
* @Date: 2021/12/6
*/
@Slf4j
@Component
@RocketMQMessageListener(topic = RocketMQConst.TOPIC_TRANSACTION, consumerGroup = RocketMQConst.GROUP_CONSUMER_TRANSACTION)
public class RocketMQTransactionConsumer implements RocketMQListener<String> {
@Override
public void onMessage(String s) {
log.info("rocketMQ consumer, topic:{}, message:{}", RocketMQConst.TOPIC_TRANSACTION, s);
}
}
4 测试
4.1 同步消息
GET 方式请求接口
http://127.0.0.1:8750/api/rocketmq/producer/send/sync?message=sync1638863656
生产者端日志
2021-12-07 15:54:16 | INFO | http-nio-8750-exec-1 | com.ljq.demo.springboot.rocketmq.producer.controller.RocketMQProducerController 60| request param: sync1638863656
2021-12-07 15:54:16 | INFO | http-nio-8750-exec-1 | com.ljq.demo.springboot.rocketmq.producer.common.rocketmq.RocketMQProducer 56| RocketMQ producer, sync message, topic: rocketmq_topic_sync, message: sync1638863656
2021-12-07 15:54:16 | DEBUG | http-nio-8750-exec-1 | org.apache.rocketmq.spring.core.RocketMQTemplate 561| send message cost: 209 ms, msgId:7F00000151D818B4AAC2224FD39C0000
2021-12-07 15:54:16 | INFO | http-nio-8750-exec-1 | com.ljq.demo.springboot.rocketmq.producer.controller.RocketMQProducerController 62| sendResult: SendResult [sendStatus=SEND_OK, msgId=7F00000151D818B4AAC2224FD39C0000, offsetMsgId=B63D13EF00002A9F000000000002B306, messageQueue=MessageQueue [topic=rocketmq_topic_sync, brokerName=broker-a, queueId=0], queueOffset=1]
消费者端日志
2021-12-07 15:54:16 | DEBUG | ConsumeMessageThread_1 | org.apache.rocketmq.spring.support.DefaultRocketMQListenerContainer 356| received msg: MessageExt [brokerName=broker-a, queueId=0, storeSize=289, queueOffset=1, sysFlag=0, bornTimestamp=1638863656862, bornHost=/producer_public_ip:25530, storeTimestamp=1638863656093, storeHost=/broker_public_ip:10911, msgId=B63D13EF00002A9F000000000002B306, commitLogOffset=176902, bodyCRC=1374578763, reconsumeTimes=0, preparedTransactionOffset=0, toString()=Message{topic='rocketmq_topic_sync', flag=0, properties={MIN_OFFSET=0, MAX_OFFSET=2, CONSUME_START_TIME=1638863656905, id=72c62515-2696-c417-b471-d0e743b14700, UNIQ_KEY=7F00000151D818B4AAC2224FD39C0000, CLUSTER=DefaultCluster, contentType=text/plain;charset=UTF-8, timestamp=1638863656680}, body=[115, 121, 110, 99, 49, 54, 51, 56, 56, 54, 51, 54, 53, 54], transactionId='null'}]
2021-12-07 15:54:16 | INFO | ConsumeMessageThread_1 | com.ljq.springboot.rocketmq.consumer.common.rocketmq.RocketMQSyncConsumer 22| rocketMQ consumer, topic:rocketmq_topic_sync, message:sync1638863656
2021-12-07 15:54:17 | DEBUG | ConsumeMessageThread_1 | org.apache.rocketmq.spring.support.DefaultRocketMQListenerContainer 361| consume 7F00000151D818B4AAC2224FD39C0000 cost: 0 ms
4.2 异步消息
GET 方式请求接口
http://127.0.0.1:8750/api/rocketmq/producer/send/async?message=async1638863774
生产者端日志
2021-12-07 15:56:13 | INFO | http-nio-8750-exec-4 | com.ljq.demo.springboot.rocketmq.producer.controller.RocketMQProducerController 74| request param: async1638863774
2021-12-07 15:56:13 | INFO | http-nio-8750-exec-4 | com.ljq.demo.springboot.rocketmq.producer.common.rocketmq.RocketMQProducer 66| RocketMQ producer, async message, topic: rocketmq_topic_async, message: async1638863774
2021-12-07 15:56:13 | INFO | NettyClientPublicExecutor_2 | com.ljq.demo.springboot.rocketmq.producer.common.rocketmq.RocketMQProducer 70| async message send success !!!
topic: rocketmq_topic_async, message: async1638863774, result: SendResult [sendStatus=SEND_OK, msgId=7F00000151D818B4AAC222519C4E0002, offsetMsgId=B63D13EF00002A9F000000000002B54A, messageQueue=MessageQueue [topic=rocketmq_topic_async, brokerName=broker-a, queueId=1], queueOffset=3]
消费者端日志
2021-12-07 15:56:13 | DEBUG | ConsumeMessageThread_2 | org.apache.rocketmq.spring.support.DefaultRocketMQListenerContainer 356| received msg: MessageExt [brokerName=broker-a, queueId=1, storeSize=291, queueOffset=3, sysFlag=0, bornTimestamp=1638863773774, bornHost=/producer_public_ip:25530, storeTimestamp=1638863773052, storeHost=/broker_public_ip:10911, msgId=B63D13EF00002A9F000000000002B54A, commitLogOffset=177482, bodyCRC=1908008441, reconsumeTimes=0, preparedTransactionOffset=0, toString()=Message{topic='rocketmq_topic_async', flag=0, properties={MIN_OFFSET=0, MAX_OFFSET=4, CONSUME_START_TIME=1638863773850, id=2ba05a08-1e77-2307-3ead-ca4f10b9087e, UNIQ_KEY=7F00000151D818B4AAC222519C4E0002, CLUSTER=DefaultCluster, contentType=text/plain;charset=UTF-8, timestamp=1638863773773}, body=[97, 115, 121, 110, 99, 49, 54, 51, 56, 56, 54, 51, 55, 55, 52], transactionId='null'}]
2021-12-07 15:56:13 | INFO | ConsumeMessageThread_2 | com.ljq.springboot.rocketmq.consumer.common.rocketmq.RocketMQAsyncConsumer 22| rocketMQ consumer, topic:rocketmq_topic_async, message:async1638863774
2021-12-07 15:56:13 | DEBUG | ConsumeMessageThread_2 | org.apache.rocketmq.spring.support.DefaultRocketMQListenerContainer 361| consume 7F00000151D818B4AAC222519C4E0002 cost: 0 ms
4.3 事务消息
GET 方式请求接口
http://127.0.0.1:8750/api/rocketmq/producer/send/transaction?message=transaction_success1638863846
生产者端日志
2021-12-07 15:57:25 | INFO | http-nio-8750-exec-6 | com.ljq.demo.springboot.rocketmq.producer.controller.RocketMQProducerController 87| request param: transaction_success1638863846
2021-12-07 15:57:25 | INFO | http-nio-8750-exec-6 | com.ljq.demo.springboot.rocketmq.producer.common.rocketmq.RocketMQProducer 89| {RocketMQ producer, transaction message, topic: rocketmq_topic_transaction, message: transaction_success1638863846, transactionId: bd81c15f-cea8-4301-93ff-2b67d9ebf777}
2021-12-07 15:57:25 | INFO | http-nio-8750-exec-6 | com.ljq.demo.springboot.rocketmq.producer.common.rocketmq.RocketMQTransactionListenerImpl 29| RocketMQ local transaction execute, message:[116, 114, 97, 110, 115, 97, 99, 116, 105, 111, 110, 95, 115, 117, 99, 99, 101, 115, 115, 49, 54, 51, 56, 56, 54, 51, 56, 52, 54],transactionId: bd81c15f-cea8-4301-93ff-2b67d9ebf777, param:transaction_success1638863846
2021-12-07 15:57:25 | INFO | http-nio-8750-exec-6 | com.ljq.demo.springboot.rocketmq.producer.common.rocketmq.RocketMQTransactionListenerImpl 34| transactionParam:transaction_success1638863846
2021-12-07 15:57:25 | INFO | http-nio-8750-exec-6 | com.ljq.demo.springboot.rocketmq.producer.common.rocketmq.RocketMQTransactionListenerImpl 37| local transaction handle result: success
2021-12-07 15:57:25 | INFO | http-nio-8750-exec-6 | com.ljq.demo.springboot.rocketmq.producer.controller.RocketMQProducerController 89| transactionSendResult: SendResult [sendStatus=SEND_OK, msgId=7F00000151D818B4AAC22252B5200003, offsetMsgId=null, messageQueue=MessageQueue [topic=rocketmq_topic_transaction, brokerName=broker-a, queueId=3], queueOffset=290]
消费者端日志
2021-12-07 15:57:25 | DEBUG | ConsumeMessageThread_1 | org.apache.rocketmq.spring.support.DefaultRocketMQListenerContainer 356| received msg: MessageExt [brokerName=broker-a, queueId=3, storeSize=468, queueOffset=7, sysFlag=8, bornTimestamp=1638863845664, bornHost=/producer_public_ip:25530, storeTimestamp=1638863845020, storeHost=/broker_public_ip:10911, msgId=B63D13EF00002A9F000000000002B83F, commitLogOffset=178239, bodyCRC=537187155, reconsumeTimes=0, preparedTransactionOffset=177773, toString()=Message{topic='rocketmq_topic_transaction', flag=0, properties={TRAN_MSG=true, CONSUME_START_TIME=1638863845813, MIN_OFFSET=0, REAL_TOPIC=rocketmq_topic_transaction, MAX_OFFSET=8, id=cd55c971-6d35-5db1-d124-4ed96bfc5180, UNIQ_KEY=7F00000151D818B4AAC22252B5200003, CLUSTER=DefaultCluster, TRANSACTION_ID=bd81c15f-cea8-4301-93ff-2b67d9ebf777, contentType=text/plain;charset=UTF-8, PGROUP=rocketmq-producer-group, WAIT=false, timestamp=1638863845599, REAL_QID=3}, body=[116, 114, 97, 110, 115, 97, 99, 116, 105, 111, 110, 95, 115, 117, 99, 99, 101, 115, 115, 49, 54, 51, 56, 56, 54, 51, 56, 52, 54], transactionId='7F00000151D818B4AAC22252B5200003'}]
2021-12-07 15:57:25 | INFO | ConsumeMessageThread_1 | com.ljq.springboot.rocketmq.consumer.common.rocketmq.RocketMQTransactionConsumer 21| rocketMQ consumer, topic:rocketmq_topic_transaction, message:transaction_success1638863846
2021-12-07 15:57:25 | DEBUG | ConsumeMessageThread_1 | org.apache.rocketmq.spring.support.DefaultRocketMQListenerContainer 361| consume 7F00000151D818B4AAC22252B5200003 cost: 0 ms
5 注意事项
- 事务消息: RocketMQ 在事务消息处理时,从
executeLocalTransaction
方法中获取到的消息体为二进制数据/byte 数组,并非原始数据,在发送事务消息的时候,也可以通过param
参数将传递业务数据,org.apache.rocketmq.spring.core.RocketMQTemplate#sendMessageInTransaction
方法的最后一个参数即为executeLocalTransaction
方法消息之外的入参 - 事务消息参数配置: 当事务消息状态为
unkonwn
时,RocketMQ 会对消息进行事务状态检查,当达到最大检查次数并且事务仍然为unknown
时,消息就会自动回滚
// 事务消息检查最大次数
private int transactionCheckMax = 15;
// 事务消息检查时间间隔
private long transactionCheckInterval = 60 * 1000;
配置文件
./conf/broker.conf
如需改动则需要手动添加配置