认识RabbitMq 第二篇 (交换机,死信队列,延迟队列)

交换机

简单介绍

上一篇博客都是创建了一个工作队列,假设的是工作队列背后,每个任务都恰好交付给一个消费者(工作线程)。在这一部分,我们将做一些完全不同的事情。我们将消息传递给多个消费者,这种模式就是 “发布、订阅”
简单实验
构建一个简单的日志系统,将两个程序组成:第一个程序将发出日志消费,第二个程序是消费者,其中我们会启动两个消费者,其中一个消费者接收到消息后把日志存储早磁盘。另外一个消费者接收到消息后把消息打印出来 ,事实上第一个程序发出的日志消息将广播给所有消费者。

Exchanges 概念

RabbitMQ 消息传递模型的核心思想是:生产者的消息从不会直接发送到队列 实际上,通常生产者甚至都不知道这些消息传递到了那些队列中。
生产者只能将消息发送到交换机 ,交换机的工作非常简单。接收来自生产者的消息。推入队列,交换机必须知道如何处理收到的消息,应该把这些消息放到特定队列。都是由交换机的类型决定的。

交换机的类型

直接,主题,标题,扇出

无名(Exchange)

之前上篇博客中我们对于交换机的部分都是一个空的字符串

在这里插入图片描述

临时队列

每当我们链接Rabbit 的时候 我们都需要一个全新的空的队列,为此可以创建一个具有随机名称的队列,或者能让服务器为我们选择一个随机队列名称就好了,其次一旦我们断开了消费者的链接,队列就会被自动删除。
创建临时队列方式

String queueName = channel.queneDeclare().getQuene();

Fanout (扇出)

Fanout 这种类型非常简单 正如从名称中猜到的那样 它将接收到的所有消息广播到它知道的所有队列中,系统中默认有些 exchange 类型。

消费者代码 01
// 消息接收
public class ReceiveLogs01 {
   
    // 交换机名称
    public static final String EXCHANGE_NAME ="logs";
    public static void main(String[] args) throws Exception{
   
    // 创建链接
        Channel channel = RabbitmqUtil.getChannel();
        // 创建一个交换机
        channel.exchangeDeclare(EXCHANGE_NAME,"fanout");
        // 生成一盒临时的队列
        String queueName = channel.queueDeclare().getQueue();
        // 绑定交换机与队列
        channel.queueBind(queueName,EXCHANGE_NAME,"");
        System.out.println("等待接收消息");
        // 接收消息
        DeliverCallback deliverCallback = (consumerTag,message) ->{
   
            System.out.println("02 new String(message.getBody(),\"UTF-8\") = " + new String(message.getBody(),"UTF-8"));
        };
        channel.basicConsume(queueName,true,deliverCallback,consumerTag ->{
   });
    }
}

消费者一和二是一样的

生产者代码
public class LOgs {
   
    // 交换机名称
    public static final String EXCHANGE_NAME ="logs";

    public static void main(String[] args) throws Exception{
   
         // 创建连接的对象
        Channel channel = RabbitmqUtil.getChannel();
        // 交换机
        channel.exchangeDeclare(EXCHANGE_NAME,"fanout");
        Scanner scanner = new Scanner(System.in);
        while (scanner.hasNext()) {
   
            String message = scanner.next();
            channel.basicPublish(EXCHANGE_NAME,"",null,message.getBytes("UTF-8"));
            System.out.println("message = " + message);

        }
    }
}

效果

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
这样一个消息可以被2个不同的消费者获取。

Direct exchange

小总结
上面构建一个简单的日志系统。可以接收许多接受者广播的日志消息。但是这个交换机的可以只让某个消费者订阅发布的部分内容。加入说我们只想吧错误的定向存储到日志文件 同时任然能够打印出所有的信息。

Direct exchange 介绍

这个就可以很好的把我们日志消息写入磁盘的程序,接收严重的错误(errors)不存储那些警告或者其他的日志消息写入磁盘空间,Fanout 这种交换并不能给我们带来很大的灵活性 他只能进行无意识的广播,在这里我们将使用 direct 这种类型来进行替换 这种类型的工作方式是 消息只去到它绑定的 routingkey 队列中去

代码

**这个负责 info 和 警告 **

public class Direct {
   
    // 交换机的名字
    public static final String EXCHANGE_NAME   = "DirectLogs";

    public static void main(String[] args) throws Exception {
   
        // 获取到 信道
        Channel channel = RabbitmqUtil.getChannel();
        // 声明交换机
        channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.DIRECT);
        // 声明一个队列
        channel.queueDeclare("console",false,false,false,null);

        channel.queueBind("console",EXCHANGE_NAME,"info");
          channel.queueBind("console",EXCHANGE_NAME,"warning");
        // 接收消息
        DeliverCallback deliverCallback = (consumerTag,message) ->{
   
            System.out.println("new String(message.getBody(),\"UTF-8\") = " + new String(message.getBody(),"UTF-8"));
        };
        // 消费者取消消息时回调接口
        channel.basicConsume("cosole",true,deliverCallback,consumerTag ->{
   });
    }
}

这个是 错误



public class Direct_1 {
   
    // 交换机的名字
    public static final String EXCHANGE_NAME   = "DirectLogs";

    public static void main(String[] args) throws Exception {
   
        // 获取到 信道
        Channel channel = RabbitmqUtil.getChannel();
        // 声明交换机
        channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.DIRECT);
        // 声明一个队列
        channel.queueDeclare("disk",false,false,false,null);

        channel.queueBind("disk"
<think>好的,用户想在Spring Boot中使用RabbitMQ实现延迟队列。我需要结合之前的引用内容来回答。首先,根据引用[1][2],延迟队列可以通过TTL死信队列实现,或者使用插件。引用[4]提到插件可以解决消息堆积时第一个消息延迟影响后续的问题。所以最佳实践应该包括这两种方法,并比较优缺点。 步骤一,添加依赖。引用[1][5]提到需要添加Spring BootRabbitMQ的依赖,所以应该指导用户添加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`进行接口测试,观察消息延迟投递效果。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值