RocketMQ详解

        RocketMQ 是一个分布式消息中间件,具有高吞吐量、低延迟、可靠性高等特点。它广泛应用于大规模分布式系统中,用于解耦服务之间的直接调用、异步处理、流量削峰等场景。


一、关键组件

NameServer
        作用:轻量级服务发现组件,管理集群的元数据(Broker 地址、Topic 路由信息)。

Broker
        作用:消息存储与转发核心节点,负责消息持久化、投递和查询。

Producer
        特点:支持同步/异步/单向发送,可指定消息队列(MessageQueue)或由负载均衡策略选择。
        分组(Producer Group):用于事务消息,同一组的生产者需保证事务一致性。

Consumer
        分组(Consumer Group):多个消费者共同消费同一 Topic,实现负载均衡。
        消费模式:
                集群消费:同组消费者分摊消息。
                广播消费:每个消费者收到全量消息。
        拉取/推送模型:底层为拉取(Pull),封装为推送(Push)接口。

Topic
        逻辑消息分类,可跨多个 Broker 分布。

Queue

        是并行存储和消费的最小单元。

其他组件
        Filter Server:消息过滤(基于 SQL 表达式或 Tag)。
        事务消息:二阶段提交机制,支持分布式事务。


二、Topic、GroupName 和 Tag 

       在 Apache RocketMQ 中,Topic、GroupName 和 Tag 是用于管理和组织消息传递的关键概念,各自承担着不同的角色:

        Topic

        定义:Topic 是消息的一级分类方法。每个消息都必须指定一个 Topic,类似于消息的主题或者类别。

        作用:生产者将消息发送到特定的 Topic,消费者则订阅自己指定的 Topic 来接收消息。这样可以实现消息的发布/订阅模式,使得消息的生产和消费解耦。

        GroupName

        生产者 Group 是用于标识一组生产者实例的名称。同一个生产者 Group 中的生产者实例共享相同的配置和状态。如果不指定生产者组,大多数版本的RocketMQ 会使用默认的组名(如 DEFAULT_PRODUCER)。

(1)标识生产者身份

        生产者 Group 是 RocketMQ 中唯一标识一组生产者实例的名称。

        通过生产者 Group,RocketMQ 可以管理和监控生产者的状态。

(2)事务消息支持

        如果使用 RocketMQ 的事务消息功能,生产者 Group 是必须的。

        RocketMQ 会根据生产者 Group 来回查事务状态(即检查本地事务是否执行成功)。

(3)消息重试

        生产者 Group 与消息的重试机制相关。如果消息发送失败,RocketMQ 会根据生产者 Group 的配置进行重试。

(4)监控和管理

         在 RocketMQ 控制台或监控系统中,生产者 Group 用于区分不同的生产者实例,方便运维和监控。

        消费者 Group 是用于标识一组消费者实例的名称。同一个消费者 Group 中的消费者实例共享相同的订阅配置和消费进度。

(1)标识消费者身份

        消费者 Group 是 RocketMQ 中唯一标识一组消费者实例的名称。

        通过消费者 Group,RocketMQ 可以管理和监控消费者的状态。

(2)消费模式

        消费者 Group 决定了消息的消费模式:

        集群模式(CLUSTERING):同一个消费者 Group 中的多个消费者实例共同消费消息,每条消息只会被一个消费者实例消费。

         广播模式(BROADCASTING):同一个消费者 Group 中的每个消费者实例都会消费所有的消息。

(3)消费进度管理

        RocketMQ 会为每个消费者 Group 维护消费进度(Offset),确保消息不会被重复消费或丢失。

(4)监控和管理

        在 RocketMQ 控制台或监控系统中,消费者 Group 用于区分不同的消费者实例,方便运维和监控。

        Tag

        定义:Tag 是附加在消息上的二级分类标签,允许对 Topic 下的消息做进一步细分。

        作用:消费者可以根据需要订阅 Topic 下带有特定 Tag 的消息。这提供了一种更细粒度的消息过滤机制,使得消费者能够更加精确地控制它们想要接收的消息类型。


三、Rocket MQ初始化

1.安装并启动RocketMQ

本地部署 RocketMQ | RocketMQ

2.引入依赖

        在每个微服务的pom.xml加入依赖

<dependency>
    <groupId>org.apache.rocketmq</groupId>
    <artifactId>rocketmq-spring-boot-starter</artifactId>
    <version>2.2.3</version>
</dependency>


四、MQ生产者初始化

1.配置文件
server:
  port: 8080
 
rocketmq:
  name-server: 127.0.0.1:9876
  producer:
    group: demo_producer_group_V3
    max-message-size: 4194304 # 4MB
    send-message-timeout: 2000
    retry-times-when-send-failed: 1
    retry-times-when-send-async-failed: 1
2.配置类(RocketMQ 生产者的 Spring Bean)
@Getter
@Setter
@ToString
@Configuration
@ConfigurationProperties(prefix = "rocketmq.producer")
public class MQProducerConfigure {
    private String group; // 生产者的group
    private String nameServer; // 与配置文件中的 name-server 匹配
    private int maxMessageSize = 4194304; // 默认值 4MB
    private int sendMessageTimeout = 3000; // 默认值 3秒
    private int retryTimesWhenSendFailed = 2; // 默认值 2次
    private int retryTimesWhenSendAsyncFailed = 2; // 默认值 2次

    @Bean
    public DefaultMQProducer defaultProducer() throws MQClientException {
        DefaultMQProducer producer = new DefaultMQProducer(group);
        producer.setNamesrvAddr(nameServer);
        producer.setMaxMessageSize(maxMessageSize);
        producer.setSendMsgTimeout(sendMessageTimeout);
        producer.setRetryTimesWhenSendFailed(retryTimesWhenSendFailed);
        producer.setRetryTimesWhenSendAsyncFailed(retryTimesWhenSendAsyncFailed);
        producer.start();
        return producer;
    }
}

        如果你的项目需要需要不同的生产者组来处理不同的业务逻辑,可以配置多个 DefaultMQProducer Bean,用:@Bean("producer1"),每个 Bean 使用不同的 group。

//发送配置类追加Bean
@Bean("producer2")
public DefaultMQProducer producer2() throws MQClientException {
    DefaultMQProducer producer = new DefaultMQProducer("ProducerGroup2");
    //......
    producer.start();
    return producer;
}


//MQ发送类使用@Qualifier区分Bean
@Autowired
@Qualifier("producer2")
private DefaultMQProducer producer2;
//发送逻辑


五、MQ消费者初始化

1.配置文件
server:
  port: 8081

rocketmq:
  consumer:
    group-name: your_consumer_group
    namesrv-addr: 127.0.0.1:9876
    topic: TestTopic
    tag: TestTag
    consume-message-batch-max-size: 10
2.配置类

        配置类目前的设计是针对 topics 的配置格式相同的 topics 进行订阅的,如果你需要支持多个不同的 topic和group,可以采用在配置类里面为每个编写单独的配置Bean。

        被@Bean注解标注的消费者方法会在Spring容器初始化期间被执行,而其中关键是.start()方法启动了消费者进程,使得接下来的流程将进入消息消费的循环中。

RocketMQ 支持两种消费模式:

集群模式(CLUSTERING)默认模式。

        同一个消费者组(Consumer Group)中的多个消费者共同消费同一个 Topic 的消息。

        每条消息只会被消费者组中的一个消费者消费。

        适合负载均衡场景。

广播模式(BROADCASTING)

        同一个消费者组中的每个消费者都会消费所有的消息。

        每条消息会被消费者组中的每个消费者消费一次。

        适合消息需要广播到所有消费者的场景。

        

设置:

        consumer.setMessageModel(MessageModel.CLUSTERING); // 集群模式
        consumer.setMessageModel(MessageModel.BROADCASTING); // 广播模式

@Getter
@Setter
@ToString
@Configuration
@ConfigurationProperties(prefix = "rocketmq.consumer")
public class MQConsumerOneConfigure {
    private String groupName;
    private String namesrvAddr;
    private String topic;
    private String tag;
    private Integer consumeMessageBatchMaxSize = 1; // 默认每次消费一条消息


    @Bean("defaultMQPushConsumer")
    public DefaultMQPushConsumer defaultConsumer() throws MQClientException {
        // 创建消费者实例
        DefaultMQPushConsumer consumer = new DefaultMQPushConsumer(groupName);
        consumer.setNamesrvAddr(namesrvAddr);
        consumer.setConsumeThreadMin(5);
        consumer.setConsumeThreadMax(20);
        consumer.setConsumeMessageBatchMaxSize(consumeMessageBatchMaxSize);
        consumer.registerMessageListener(new MQConsumeMsgListenerProcessor());
        consumer.setConsumeFromWhere(ConsumeFromWhere.CONSUME_FROM_LAST_OFFSET);
        // 订阅主题并启动消费者
        try {
            //void subscribe(String topic, String tag);
            consumer.subscribe(topic,tag);
            consumer.start();
        } catch (MQClientException e) {
            throw new RuntimeException("Failed to start RocketMQ consumer", e);
        }
        return consumer;
    }
}
3.监听器方法
@Component
public class MQConsumeMsgListenerProcessor implements MessageListenerConcurrently {
    @Override
    public ConsumeConcurrentlyStatus consumeMessage(List<MessageExt> msgList, ConsumeConcurrentlyContext consumeConcurrentlyContext) {
        if (CollectionUtils.isEmpty(msgList)) {
            return ConsumeConcurrentlyStatus.CONSUME_SUCCESS;
        }
        MessageExt messageExt = msgList.get(0);
        try {
            String topic = messageExt.getTopic();
            String tags = messageExt.getTags();
            String body = new String(messageExt.getBody(), "utf-8");
        } catch (Exception e) {
            // TODO 异常处理逻辑
        }
        // 手动 ACK
        return ConsumeConcurrentlyStatus.CONSUME_SUCCESS;
    }
}


六、MQ生产者发送

1.同步消息

        生产者阻塞等待Broker返回发送结果(SendResult)。这种模式提供了较高的可靠性,但性能较低,因为它涉及到网络往返延迟。

    /*通过 @ConfigurationProperties 注解,
    可以将配置文件中的属性绑定到 MQProducerConfigure 类的字段上,
    然后通过 @Bean 方法创建并配置 DefaultMQProducer 实例。
     */
    @Autowired
    private DefaultMQProducer defaultMQProducer;


    @GetMapping("/send")
    public SendResult send(String msg) throws InterruptedException, RemotingException, MQClientException, MQBrokerException {
        if (StringUtils.isEmpty(msg)) {
            return null;
        }
        LOGGER.info("发送MQ消息内容:" + msg);
        Message sendMsg = new Message("TestTopic", "TestTag", msg.getBytes());
        // 默认3秒超时
        SendResult sendResult = defaultMQProducer.send(sendMsg);
        LOGGER.info("消息发送响应:" + sendResult.toString());
        return sendResult;
    }
2.异步消息

        生产者发送消息后不需要等待Broker的响应,而是提供一个回调函数,在消息发送成功或失败时会被调用。这种方式提高了吞吐量,适用于对响应时间敏感的应用场景。

        defaultMQProducer.send(sendMsg, new SendCallback() {
            @Override
            public void onSuccess(SendResult sendResult) {
                System.out.printf("消息发送成功: %s%n", sendResult);
            }

            @Override
            public void onException(Throwable e) {
                System.err.println("消息发送失败: " + e.getMessage());
            }
        });
3.单向消息

        生产者只负责发送消息,不关心消息是否成功送达。这种方式适用于某些日志收集等对丢失消息容忍度较高的场景。

        defaultMQProducer.sendOneway(sendMsg);
4.批量消息
        List<Message> messages = new ArrayList<>();
        for (int i = 0; i < 10; i++) {
            Message sendMsg = new Message("TestTopic", "TestTag",msg.getBytes(RemotingHelper.DEFAULT_CHARSET));
            messages.add(sendMsg);
        }
        // 批量发送消息
        defaultMQProducer.send(messages);


七、RocketMQTempate简化操作

简化配置
通过 Spring Boot 的自动配置,无需手动创建 DefaultMQProducer 和 DefaultMQPushConsumer。

易用性
提供了更简洁的 API,支持同步、异步、单向、延迟、批量等多种消息发送方式。

与 Spring 生态集成
支持 Spring 的依赖注入、事务管理、AOP 等特性。

自动序列化和反序列化
支持将 Java 对象自动序列化为消息体,以及将消息体反序列化为 Java 对象。

1.依赖注入
<dependency>
    <groupId>org.apache.rocketmq</groupId>
    <artifactId>rocketmq-spring-boot-starter</artifactId>
    <version>2.2.2</version> <!-- 使用最新版本 -->
</dependency>
2.配置文件

(1)生产者服务配置

# application.yml 示例
rocketmq:
    name-server: 127.0.0.1:9876 # 替换为你的NameServer地址
    producer:
        group: your-producer-group
        send-message-timeout: 3000 # 发送超时时间,默认3000ms
        retry-times-when-send-failed: 2 # 发送失败时的重试次数

(2)消费者服务配置

# application.yml 示例
rocketmq:
    name-server: 127.0.0.1:9876 # 替换为你的NameServer地址
    consumer:
        consume-thread-min: 5 # 最小消费线程数
        consume-thread-max: 10 # 最大消费线程数
        namesrvAddr: 127.0.0.1:9876 # 如果没有全局设置,则可以在这里单独设置NameServer地址
        # broadcast: true # 默认false, 表示集群消费模式;如果设为true则表示广播消费模式
3.代码示例

(1)生产者代码示例

同步消息

@Service
public class RocketMQService {

    @Autowired
    private RocketMQTemplate rocketMQTemplate;

    public void sendMessage(String topic, User userMessage) {
        rocketMQTemplate.convertAndSend(topic, userMessage);
    }
}

异步消息

rocketMQTemplate.asyncSend("your-topic", userMessage, new SendCallback() {
    @Override
    public void onSuccess(SendResult sendResult) {
        // 成功回调处理
    }

    @Override
    public void onException(Throwable throwable) {
        // 异常回调处理
    }
});

单向消息

rocketMQTemplate.sendOneWay("your-topic", "your-message");

(2)消费者代码示例

        @RocketMQMessageListener可以追加selecrorExpression实现对tag的过滤

@Service
@RocketMQMessageListener(topic = "your-topic", consumerGroup = "your-consumer-group")
public class YourMessageListener implements RocketMQListener<User> {

    @Override
    public void onMessage(User userMessage) {
        // 处理接收到的消息
    }
}
4.生产者多个Group

        在 RocketMQ 中,RocketMQTemplate 的 group 是通过配置文件(如 application.yml)指定的,默认情况下,一个 RocketMQTemplate 实例只需绑定一个生产者 Group。如果你需要支持多个生产者 Group,可以通过以下方式实现:

(1)定义多个 RocketMQTemplate Bean
        每个 RocketMQTemplate 实例绑定一个生产者 Group。

rocketmq:
  name-server: localhost:9876
  producer:
    group1:
      group: ProducerGroup1 # 生产者 Group1
      send-message-timeout: 3000
      retry-times-when-send-failed: 2
    group2:
      group: ProducerGroup2 # 生产者 Group2
      send-message-timeout: 5000
      retry-times-when-send-failed: 3
@Configuration
@ConfigurationProperties(prefix = "rocketmq.producer")
public class RocketMQConfig {

    private Map<String, ProducerConfig> group1 = new HashMap<>();
    private Map<String, ProducerConfig> group2 = new HashMap<>();

    public Map<String, ProducerConfig> getGroup1() {
        return group1;
    }

    public void setGroup1(Map<String, ProducerConfig> group1) {
        this.group1 = group1;
    }

    public Map<String, ProducerConfig> getGroup2() {
        return group2;
    }

    public void setGroup2(Map<String, ProducerConfig> group2) {
        this.group2 = group2;
    }

    @Bean("rocketMQTemplate1")
    public RocketMQTemplate rocketMQTemplate1() {
        RocketMQTemplate template = new RocketMQTemplate();
        template.setProducerGroup(group1.get("group")); // 动态注入 Group1
        template.setNameServer("localhost:9876");
        return template;
    }

    @Bean("rocketMQTemplate2")
    public RocketMQTemplate rocketMQTemplate2() {
        RocketMQTemplate template = new RocketMQTemplate();
        template.setProducerGroup(group2.get("group")); // 动态注入 Group2
        template.setNameServer("localhost:9876");
        return template;
    }

    public static class ProducerConfig {
        private String group;
        private int sendMessageTimeout;
        private int retryTimesWhenSendFailed;

        // Getters and Setters
        public String getGroup() {
            return group;
        }

        public void setGroup(String group) {
            this.group = group;
        }

        public int getSendMessageTimeout() {
            return sendMessageTimeout;
        }

        public void setSendMessageTimeout(int sendMessageTimeout) {
            this.sendMessageTimeout = sendMessageTimeout;
        }

        public int getRetryTimesWhenSendFailed() {
            return retryTimesWhenSendFailed;
        }

        public void setRetryTimesWhenSendFailed(int retryTimesWhenSendFailed) {
            this.retryTimesWhenSendFailed = retryTimesWhenSendFailed;
        }
    }
}

(2)在代码中通过 @Qualifier 注入不同的 RocketMQTemplate&动态切换生产者 Group
        发送消息时,选择对应的 RocketMQTemplate。

@Service
public class MessageService {

    @Autowired
    @Qualifier("rocketMQTemplate1")
    private RocketMQTemplate rocketMQTemplate1;

    @Autowired
    @Qualifier("rocketMQTemplate2")
    private RocketMQTemplate rocketMQTemplate2;

    /**
     * 使用 ProducerGroup1 发送消息
     */
    public void sendToGroup1(String topic, String tag, String message) {
        rocketMQTemplate1.syncSend(topic + ":" + tag, message);
    }

    /**
     * 使用 ProducerGroup2 发送消息
     */
    public void sendToGroup2(String topic, String tag, String message) {
        rocketMQTemplate2.syncSend(topic + ":" + tag, message);
    }
}


八、MQ分区顺序消费

        生产者使用 MessageQueueSelector 指定队列选择策略,将同一业务ID(如订单ID)的消息发送到同一个 MessageQueue。消费者注册 MessageListenerOrderly 监听器,单线程处理同一队列的消息。

        添加队列:sh mqadmin updateTopic -n <namesrvAddr> -t <topicName> -r <readQueueNums> -w <writeQueueNums>

1.RocketMQTemplate实现

(1)生产者配置

        生产者需要确保同一业务ID的消息发送到同一个MessageQueue,这样才能保证顺序性。

public void sendOrderedMessage(String topic, String message, String shardingKey) {
    try {
        SendResult result = rocketMQTemplate.syncSendOrderly(topic, 
                           MessageBuilder.withPayload(message).build(), 
                           shardingKey,
                           2000); // 超时时间
        log.info("消息发送成功: {}", result);
    } catch (Exception e) {
        log.error("消息发送失败", e);
        // 根据业务需要处理失败逻辑
    }
}

(2)消费者配置

        消费者需要配置为顺序消费模式,并且正确处理消费失败的情况。

@Service
@Slf4j
@RocketMQMessageListener(
        topic = "your_topic",
        consumerGroup = "your_consumer_group",
        consumeMode = ConsumeMode.ORDERLY,  // 设置为顺序消费
        messageModel = MessageModel.CLUSTERING  // 集群模式
)
public class RocketConsumer implements RocketMQListener<String> {

    @Override
    public void onMessage(String message) {
        try {
            // 处理消息
            log.info("Received ordered message: {}", message);
            // 实际业务处理...
        } catch (BusinessException e) { // 业务异常
            log.error("业务处理失败但可重试, message: {}-{}", message, e);
            throw new RuntimeException(e); // 触发重试
        } catch (Exception e) {
            log.error("系统异常,需要人工干预, message: {}", message, e);
            // 可以将消息存入数据库或死信队列
            // 注意:这里仍要抛出异常才能保证不会ACK
            throw new RuntimeException(e);
        }
    }
}
2.原生API实现

(1)使用MessageQueueSelector去设置路由策略

    @Autowired
    private DefaultMQProducer defaultMQProducer;
    // SendResult 仅仅是消息发送的结果
    public SendResult sendOrderedMessage(String topic, String messageBody, String shardingKey) throws Exception {
        Message message = new Message(topic, messageBody.getBytes());
        // 使用 shardingKey 选择队列,确保相同 shardingKey 进入同一队列
        SendResult sendResult = defaultMQProducer.send(message, new MessageQueueSelector() {
            @Override
            public MessageQueue select(List<MessageQueue> mqs, Message msg, Object arg) {
                String key = (String) arg;
                int index = Math.abs(key.hashCode()) % mqs.size(); // 取模计算队列索引
                return mqs.get(index);
            }
        }, shardingKey); // 传入 shardingKey 作为路由依据
        return sendResult;
    }

(2)设置独立单线程的消费者配置类并加载监听器

@Getter
@Setter
@Configuration
@ConfigurationProperties(prefix = "rocketmq.consumer.orderly") // 独立配置前缀
public class OrderConsumerConfigure {

    private String groupName;
    private String namesrvAddr;
    private String topic;
    private String tag = "*";
    private Integer consumeThreadMin = 1; // 顺序消费建议单线程
    private Integer consumeThreadMax = 1; // 必须为1

    @Bean(name = "orderlyConsumer", initMethod = "start", destroyMethod = "shutdown")
    public DefaultMQPushConsumer orderlyConsumer() throws MQClientException {
        DefaultMQPushConsumer consumer = new DefaultMQPushConsumer(groupName);
        consumer.setNamesrvAddr(namesrvAddr);
        consumer.setConsumeThreadMin(consumeThreadMin);
        consumer.setConsumeThreadMax(consumeThreadMax); // 关键点:单线程
        consumer.setConsumeFromWhere(ConsumeFromWhere.CONSUME_FROM_LAST_OFFSET);
        consumer.setMessageListener(new OrderlyMessageListener()); // 使用顺序监听器
        // 订阅主题并启动消费者
        try {
            //void subscribe(String topic, String tag);
            consumer.subscribe(topic,tag);
            consumer.start();
        } catch (MQClientException e) {
            throw new RuntimeException("Failed to start RocketMQ consumer", e);
        }
        return consumer;
    }
}

(3)配置监听器 

        MessageListenerOrderly确保消息按照它们在队列中的顺序被消费。RocketMQ 将每个队列分配给一个线程进行处理,从而保证了该队列中消息的顺序性。

@Component
public class OrderlyMessageListener implements MessageListenerOrderly { // 关键接口变更

    @Override
    public ConsumeOrderlyStatus consumeMessage(List<MessageExt> messages, ConsumeOrderlyContext context) {
        // 1. 空消息判断
        if (messages == null || messages.isEmpty()) {
            return ConsumeOrderlyStatus.SUCCESS;
        }
        // 2. 顺序处理(单队列串行执行)
        MessageExt message = messages.get(0); // 顺序消费通常每次一条
        try {
            String body = new String(message.getBody(), "UTF-8");
            String shardingKey = message.getKeys(); // 生产者发送时指定的路由键

            System.out.printf("[顺序消费] QueueId=%d, ShardingKey=%s, Body=%s%n",
                    message.getQueueId(), shardingKey, body);

            // TODO: 业务处理(如订单状态机变更)
            // 注意:此处必须保证幂等性(可能因重试导致重复处理)
            return ConsumeOrderlyStatus.SUCCESS;
        } catch (Exception e) {
            System.err.println("消费失败,等待重试: " + message);
            // 挂起当前队列(同一队列的消息会阻塞,其他队列不受影响)
            return ConsumeOrderlyStatus.SUSPEND_CURRENT_QUEUE_A_MOMENT;
        }
    }
}

九、异步请求-响应模式

服务A:发送消息时设置Response Topic和CorrelationId(uuid)
服务B:消费消息后向Response Topic发送响应消息
服务A:监听响应主题,通过CorrelationId匹配响应

1.服务A代码(RocketMQTemplate实现)
@Service
@RocketMQMessageListener(topic = "response-topic", consumerGroup = "a-consumer-group")
public class RocketMqServiceA implements RocketMQListener<MessageExt> {
    @Autowired
    private RocketMQTemplate rocketMQTemplate;
    private Map<String, CompletableFuture<String>> pendingRequests = new ConcurrentHashMap<>();
    //手动调用发送消息
    public void process() {
        String request = "Request from A";
        String correlationId = UUID.randomUUID().toString();
        // 发送请求
        Message<String> message = MessageBuilder.withPayload(request)
                .setHeader(RocketMQHeaders.KEYS, correlationId)
                .setHeader("responseTopic", "response-topic") // 指定响应主题
                .build();
        rocketMQTemplate.send("request-topic", message);
        System.out.println("[A] Sent request: " + request);
        // 异步等待响应
        CompletableFuture<String> future = new CompletableFuture<>();
        pendingRequests.put(correlationId, future);
        try {
            String response = future.get(10, TimeUnit.SECONDS);
            System.out.println("[A] Received response: " + response);
        } catch (TimeoutException | InterruptedException | ExecutionException e) {
            System.out.println("[A] Response timeout");
        } finally {
            pendingRequests.remove(correlationId);
        }
    }
    @Override
    public void onMessage(MessageExt message) {
        String corrId = message.getKeys();
        String response = new String(message.getBody());
        CompletableFuture<String> future = pendingRequests.get(corrId);
        if (future != null) {
            future.complete(response);
        }
    }
}

2.服务B代码(RocketMQTemplate实现)
@Service
@RocketMQMessageListener(topic = "request-topic", consumerGroup = "b-consumer-group")
public class RocketMqServiceB implements RocketMQListener<MessageExt> {
    @Autowired
    private RocketMQTemplate rocketMQTemplate;
    @Override
    public void onMessage(MessageExt message) {
        String request = new String(message.getBody());
        System.out.println("[B] Received: " + request);
        // 处理业务逻辑
        String response = "Processed by B: " + request;
        // 获取回调信息
        String replyTo = message.getProperty("response-topic");
        String corrId = message.getKeys();
        // 发送响应
        Message<String> responseMsg = MessageBuilder.withPayload(response)
                .setHeader(RocketMQHeaders.KEYS, corrId)
                .build();
        rocketMQTemplate.send(replyTo, responseMsg);
    }
}

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值