SpringBoot的RabbitMQ消息队列: 五、第四模式"Routing"

本文详细介绍了RabbitMQ中的Routing模式,通过Direct类型的交换机实现消息的选择性接收。文章提供了完整的代码示例,包括配置类、消息发送器、接收器以及控制器等,并通过实践演示了如何根据不同路由键将消息发送到不同的队列。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Routing模式,官方的解释是Receiving messages selectively,它的结构是

消费者各自监控自己的队列;交换机确定生产者的消息放入那个队列。其实这就是用Diret类型的交换机实现



接上个博客的项目过来..包括对应的配置,

6、配置文件。增加类DirectRabbitConfig,编制代码为

package com.rabbitMq.Routing;

import org.springframework.amqp.core.Binding;  
import org.springframework.amqp.core.BindingBuilder;  
import org.springframework.amqp.core.DirectExchange;  
import org.springframework.amqp.core.Queue;  
import org.springframework.context.annotation.Bean;  
import org.springframework.context.annotation.Configuration;  
  
/**Routing 模式.配置类
 * @author Administrator
 *
 */
@Configuration  
public class DirectRabbitConfig {  
  
    @Bean  
    public Queue AMessage() {  
        return new Queue("direct.A");  
    }  
  
    @Bean  
    public Queue BMessage() {  
        return new Queue("direct.B");  
    }  
  
    @Bean  
    DirectExchange directExchange() {  
        return new DirectExchange("directExchange");  
    }  
  
    @Bean  
    Binding bindingExchangeMessage(Queue AMessage, DirectExchange exchange) {  
        return BindingBuilder.bind(AMessage).to(exchange).with("orange");  
    }  
  
    @Bean  
    Binding bindingExchangeMessageBOfBlack(Queue BMessage, DirectExchange exchange) {  
        return BindingBuilder.bind(BMessage).to(exchange).with("black");  
    }  
  
    @Bean  
    Binding bindingExchangeMessageBOfGreen(Queue BMessage, DirectExchange exchange) {  
        return BindingBuilder.bind(BMessage).to(exchange).with("green");  
    }  
  
}  

A、定于两个队列

B、定义一个Direct交换机

C、三个绑定策略

7、消息发送。增加类DirectSender,编制代码为

package com.rabbitMq.Routing;

import org.slf4j.Logger;  
import org.slf4j.LoggerFactory;  
import org.springframework.amqp.core.AmqpTemplate;  
import org.springframework.beans.factory.annotation.Autowired;  
import org.springframework.stereotype.Component;  
  
@Component  
public class DirectSender {  
  
  
    protected static Logger logger=LoggerFactory.getLogger(DirectSender.class);   
      
    @Autowired  
    private AmqpTemplate rabbitTemplate;  
    public void sendOrange() {  
        String context = "hi, i am message orange";  
        logger.debug("Sender : " + context);  
        this.rabbitTemplate.convertAndSend("directExchange", "orange", context);  
    }  
  
    public void sendBlack() {  
        String context = "hi, i am messages black";  
        logger.debug("Sender : " + context);  
        this.rabbitTemplate.convertAndSend("directExchange", "black", context);  
    }  
      
    public void sendGreen() {  
        String context = "hi, i am messages green";  
        logger.debug("Sender : " + context);  
        this.rabbitTemplate.convertAndSend("directExchange", "green", context);  
    } 
}

两个对应的执行类.

package com.rabbitMq.Routing;

import org.slf4j.Logger;  
import org.slf4j.LoggerFactory;  
import org.springframework.amqp.rabbit.annotation.RabbitHandler;  
import org.springframework.amqp.rabbit.annotation.RabbitListener;  
import org.springframework.stereotype.Component;  
  
@Component  
@RabbitListener(queues = "direct.A")  
public class DirectReceiverA {  
  
    protected static Logger logger=LoggerFactory.getLogger(DirectReceiverA.class);   
      
    @RabbitHandler  
    public void process(String message) {  
        logger.debug("direct.A Receiver  : " + message);  
    }  
  
}  


package com.rabbitMq.Routing;

import org.slf4j.Logger;  
import org.slf4j.LoggerFactory;  
import org.springframework.amqp.rabbit.annotation.RabbitHandler;  
import org.springframework.amqp.rabbit.annotation.RabbitListener;  
import org.springframework.stereotype.Component;  
  
@Component  
@RabbitListener(queues = "direct.B")  
public class DirectReceiverB {  
  
    protected static Logger logger=LoggerFactory.getLogger(DirectReceiverB.class);   
      
    @RabbitHandler  
    public void process(String message) {  
        logger.debug("direct.B Receiver   : " + message);  
    }  
  
}  
10、RestController。增加类DirectController,编制代码为 

package com.rabbitMq.Routing;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

/**感觉这个模式和第一个那个好像很像,不过区分细一点总是好的.
 * @author Administrator
 *
 */
@RestController
@RequestMapping("/routing")
public class DirectController {

	@Autowired
	private DirectSender directSender;

	@RequestMapping("/send1")
	public String send1() {
		directSender.sendOrange();
		return "send1 orange ok";
	}

	@RequestMapping("/send2")
	public String send2() {
		directSender.sendBlack();
		return "send2  black ok";
	}

	@RequestMapping("/send3")
	public String send3() {
		directSender.sendGreen();
		return "send3 green ok";
	}
}

运行看对应的输入日志

浏览器上依次访问:http://localhost:8080/routing/send2



  1. 13、小结

  2. A、在配置文件中,定义了一个DirectExchange,然后对两个队列,分别配置了绑定规则。(变更绑定规则测试时,先停止命令行的spring-boot,再删除rabbitmq management中的队列)。
  3. B、发送器,发送send1会匹配到第一个Receiver收到消息,发送send2、send3都匹配到Receiver2收到消息。 
  4.      发送器在发送消息时,使用的方法是需要传入一个特定的交换机的,以及路由规则
  5.  
  6. C、接收器,依然各自监控自己的队列; 









<think>好的,用户想在Spring Boot中使用RabbitMQ实现延迟队列。我需要结合之前的引用内容来回答。首先,根据引用[1]和[2],延迟队列可以通过TTL和死信队列实现,或者使用插件。引用[4]提到插件可以解决消息堆积时第一个消息延迟影响后续的问题。所以最佳实践应该包括这两种方法,并比较优缺点。 步骤一,添加依赖。引用[1]和[5]提到需要添加Spring Boot和RabbitMQ的依赖,所以应该指导用户添加spring-boot-starter-amqp。 步骤二,配置连接。配置RabbitMQ的连接信息,这部分应该简单,用户可能知道,但需要给出示例配置。 然后是两种方法。方法一使用死信队列,需要创建交换机和队列,设置TTL。引用[1]和[2]中的示例,需要定义正常和死信交换机,绑定队列,并设置TTL。注意在发送消息时设置TTL,或者队列级别。但引用[4]指出队列级别的TTL会有问题,如果第一个消息TTL太长会阻塞后面的,所以建议用消息级别的TTL,但同样可能存在检查机制的问题。这时候需要提到插件方法。 方法二使用插件,引用[4]说明插件可以避免消息阻塞问题。需要用户安装插件,然后定义延迟交换机,发送消息时设置延迟头。这部分需要示例代码,比如配置交换机为x-delayed-message类型,发送时添加x-delay头。 然后给出示例代码,包括配置类定义交换机和队列,生产者发送消息,消费者监听队列。同时要说明如何测试,比如用Swagger接口,如引用[4]中的/swagger-ui.html。 最后比较两种方法的优缺点。方法一无需插件但可能有问题,方法二更可靠但需要安装插件。根据引用[3],用户可能知道插件方式,但另一种方法也是可行的选择。 需要确保代码正确,比如使用@Configuration配置Bean,使用RabbitTemplate发送消息,消费者用@RabbitListener。还要注意TTL的设置方式,队列参数和消息属性的区别。 相关问题部分,可以问安装插件的步骤、TTL设置方式、死信队列原理、插件方法的优势以及替代方案。这些能帮助用户深入理解。</think>### Spring Boot整合RabbitMQ实现延迟队列的实践指南 #### 一、实现方式对比 RabbitMQ实现延迟队列主要有两种方式: 1. **TTL+死信队列**:通过消息/队列的TTL过期后进入死信队列实现[^2][^4] 2. **插件方式**:使用`rabbitmq_delayed_message_exchange`插件实现更精确控制 #### 二、实现步骤(TTL+死信队列) ##### 1. 添加依赖 ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-amqp</artifactId> </dependency> ``` ##### 2. 配置连接 ```yaml spring: rabbitmq: host: localhost port: 5672 username: guest password: guest ``` ##### 3. 队列配置类 ```java @Configuration public class RabbitConfig { // 正常交换机 public static final String EXCHANGE_NORMAL = "exchange.normal"; // 死信交换机 public static final String EXCHANGE_DLX = "exchange.dlx"; // 普通队列 public static final String QUEUE_NORMAL = "queue.normal"; // 死信队列 public static final String QUEUE_DLX = "queue.dlx"; @Bean public DirectExchange normalExchange() { return new DirectExchange(EXCHANGE_NORMAL); } @Bean public DirectExchange dlxExchange() { return new DirectExchange(EXCHANGE_DLX); } @Bean public Queue normalQueue() { return QueueBuilder.durable(QUEUE_NORMAL) .withArgument("x-dead-letter-exchange", EXCHANGE_DLX) // 绑定死信交换机 .withArgument("x-dead-letter-routing-key", "dlx.routingkey") .build(); } @Bean public Queue dlxQueue() { return new Queue(QUEUE_DLX); } @Bean public Binding normalBinding() { return BindingBuilder.bind(normalQueue()) .to(normalExchange()).with("normal.routingkey"); } @Bean public Binding dlxBinding() { return BindingBuilder.bind(dlxQueue()) .to(dlxExchange()).with("dlx.routingkey"); } } ``` ##### 4. 生产者示例 ```java @RestController public class ProducerController { @Autowired private RabbitTemplate rabbitTemplate; @PostMapping("/send") public void sendMessage(@RequestParam String msg, @RequestParam Long ttl) { MessageProperties props = MessagePropertiesBuilder.newInstance() .setExpiration(String.valueOf(ttl)) .build(); Message message = new Message(msg.getBytes(), props); rabbitTemplate.send(RabbitConfig.EXCHANGE_NORMAL, "normal.routingkey", message); } } ``` ##### 5. 消费者示例 ```java @Component public class DLXConsumer { @RabbitListener(queues = RabbitConfig.QUEUE_DLX) public void processDLXMessage(Message message) { System.out.println("收到延迟消息: " + new String(message.getBody())); } } ``` #### 三、插件方式实现(推荐) ##### 1. 安装插件 ```bash rabbitmq-plugins enable rabbitmq_delayed_message_exchange ``` ##### 2. 配置延迟交换机 ```java @Bean public CustomExchange delayedExchange() { Map<String, Object> args = new HashMap<>(); args.put("x-delayed-type", "direct"); return new CustomExchange("delayed.exchange", "x-delayed-message", true, false, args); } ``` ##### 3. 发送延迟消息 ```java rabbitTemplate.convertAndSend("delayed.exchange", "routingkey", message, msg -> { msg.getMessageProperties().setHeader("x-delay", 5000); // 5秒延迟 return msg; }); ``` #### 四、两种方式对比 | 特性 | TTL+死信队列 | 插件方式 | |---------------------|-----------------------------|----------------------------| | 安装要求 | 无需插件 | 需要安装插件 | | 延迟精度 | 队列级别控制 | 消息级别控制 | | 消息阻塞问题 | 可能存在 | 完全避免 | | 最大延迟时间 | 约49天(RabbitMQ限制) | 同左 | #### 、测试建议 通过Swagger界面访问`http://localhost:8080/swagger-ui.html`进行接口测试,观察消息延迟投递效果。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值