RabbitMQ--死信队列/延迟队列--使用/原理

本文介绍了RabbitMQ中的死信队列(DLX)和延迟队列的原理和使用方法,包括消息何时会变成死信、如何配置DLX以及DLX如何与TTL配合实现延迟队列功能。此外,还探讨了延迟队列在实际应用中的场景,如订单超时和定时任务等。

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

RabbitMQ--死信队列/延迟队列--使用/原理


简介 本文介绍RabbitMQ的死信队列和延迟队列。

本内容也是Java后端面试中常见的问题。

死信队列

简介

DLX,全称为Dead-Letter-Exchange,可以称之为死信交换器,也有人称之为死信邮箱。当消息在一个队列中变成死信(dead message)之后,它能被重新被发送到另一个交换器中,这个交换器就是DLX,绑定DLX的队列就称之为死信队列。

以下几种情况会导致消息变成死信:

消息被拒绝(Basic.Reject/Basic.Nack),并且设置requeue参数为false;

消息过期;

队列达到最大长度。

DLX是一个正常的交换器,和一般的交换器没有区别,它能在任何的队列上被指定,实际上就是设置某个队列的属性。当这个队列中存在死信时,RabbitMQ就会自动地将这个消息重新发布到设置的DLX上去,进而被路由到另一个队列,即死信队列。可以监听这个队列中的消息以进行相应的处理,这个特性与将消息的TTL设置为0配合使用可以弥补immediate参数的功能。

为队列添加DLX的方法

法1:代码方式

//创建 DLX: dlx_exchange channel.exchangeDeclare(“dlx_exchange”, “direct” ); Map<String, Object> args = new HashMap<String, Object>; args.put(“x-dead-letter-exchange”, “dlx_exchange”); //为队列myqueue添加DLX channel.queueDeclare(“myqueue”, false, false, false, args); 也可以为这个DLX指定路由键。(如果没有特殊指定,则使用原队列的路由键)

args.put(“x-dead-letter-routing-key”,“dlx-routing-key”); 法2:命令方式

rabbitmqctl set_policy DLX “.*” ‘{“dead-letter-exchange”:“dlx_exchange”}’ --apply-to queues 示例 代码

channel.exchangeDeclare(“exchange.dlx”, “direct”, true); channel.exchangeDeclare(“exchange.normal”, “fanout”, true); Map<String, Object> args = new HashMap<String, Object>(); args.put(“x-message-ttl”, 10000); args.put(“x-dead-letter-exchange” , “exchange.dlx”); args.put(“x-dead-letter-routing-key” , “routingkey”); channel.queueDeclare(“queue.normal” , true, false, false, args); channel.queueBind(“queue.normal”, “exchange.normal”, “”); channel.queueDeclare(“queue.dlx”, true, false, false, null); channel.queueBind(“queue.dlx”, “exchange.dlx” , “routingkey”); channel.basicPublish(“exchange.normal” , “rk”, MessageProperties.PERSISTENT_TEXT_PLAIN, “dlx”.getBytes());

这里创建了两个交换器exchange.normal和exchange.dlx,分别绑定两个队列queue.normal和queue.dlx。

Web管理页面结果

由下图(图1-1)的Web管理页面可以看出,两个队列都被标记了“D”,这个是durable的缩写,即设置了队列持久化。queue.normal这个队列还配置了TTL、DLX和DLK,其中DLX指的是 x-dead-letter-routing-key这个属性。

在这里插入图片描述

案例分析

参考下图(图1-2),生产者首先发送一条携带路由键为“rk”的消息,然后经过交换器exchange.normal顺利地存储到队列queue.normal中。由于队列queue.normal设置了过期时间为10s,在这10s内没有消费者消费这条消息,那么判定这条消息为过期。由于设置了DLX,过期之时,消息被丢给交换器exchange.dlx中,这时找到与exchange.dlx匹配的队列queue.dlx,最后消息被存储在queue.dk这个死信队列中。

在这里插入图片描述

对于RabbitMQ来说,DLX是一个非常有用的特性。它可以处理异常情况下,消息不能够被消费者正确消费(消费者调用了Basic.Nack或者Basic.Reject)而被置入死信队列中 的情况,后续分析程序可以通过消费这个死信队列中的内容来分析当时所遇到的异常情况,进而可以改善和优化系统。DLX配合TTL使用还可以实现延迟队列的功能,详细请看下一节。大数据培训

延迟队列

简介

延迟队列用来存放延迟消息。延迟消息:指当消息被发送以后,不想让消费者立刻拿到消息,而是等待特定时间后,消费者才能拿到这个消息进行消费。

在AMQP协议中,或者RabbitMQ本身没有直接支持延迟队列的功能,但是有两种方案来间接实现:

方案1:采用rabbitmq-delayed-message-exchange 插件实现。(RabbitMQ 3.6.x开始支持)

方案2:通过前面所介绍的DLX和TTL模拟出延迟队列的功能。

在图1-2中,不仅展示的是死信队列的用法,也是延迟队列的用法,对于queue.dlx这个死信队列来说,同样可以看作延迟队列。假设一个应用中需要将每条消息都设置为10秒的延迟, 生产者通过exchange.normal这个交换器将发送的消息存储在queue.normal这个队列中。消费者订阅的并非是queue.normal这个队列,而是queue.dlx这个队列。当消息从queue.normal这个队列中过期之后被存入queue.dlx这个队列中,消费者就恰巧消费到了延迟10秒的这条消息。

在真实应用中,对于延迟队列可以根据延迟时间的长短分为多个等级,一般分为5秒、10秒、30秒、1分钟、5分钟、10分钟、30分钟、1小时这几个维度,当然也可以再细化一下。

以下图(图2-1)为例进行说明。为简化,只设置5秒、10秒、30秒、1分钟这四个等级。根据需求的不同,生产者发送消息的时候通过设置不同的路由键,将消息发送到与交换器绑定的不同的队列中。这里队列也分别配置了DLX和相应的死信队列,当相应的消息过期时,就会转存到相应的死信队列(即延迟队列)中,这样消费者根据业务自身的情况,分别选择不同延迟等级的延迟队列进行消费。

在这里插入图片描述

使用场景

延迟队列的使用场景有很多,比如:

用户下订单场景:用户下单后有30分钟的时间支付,若30分钟内没有支付,则将这个订单取消。 方案:用户下单后将取消订单的消息发送到延迟队列,延迟时间设置为30分钟。取消订单这个消息的订阅者程序在30分钟后收到消息,判断该订单的状态是否为已支付,若还没支付,则将该订单状态设置为:已取消。 定时遥控场景:用户想用手机远程遥控家里的智能设备在指定的时间工作。

方案:假设用户想要的操作是:开启热水器。首先,将开启热水器这个消息发送到延迟队列,延迟时间设置到用户想要的时间到现在时间的差值。开启热水器这个消息的订阅者程序在指定时间收到消息,再将指令推送到智能设备。

需要注意的是,延迟队列的消息是不能取消的,解决方案是:在消费消息的时候判断这个消息对应的业务的当前状态。例如:对于取消订单来说,收到消息时,读取这个消息所对应的数据库信息,如果已经是已付款状态了,就不进行任何操作了,如果是未支付状态,则改为已取消。

转载自:https://www.cnblogs.com/msjhw/p/15571806.html

<think>嗯,用户问的是如何在Java中实现RabbitMQ死信队列,需要示例教程。首先,我得回忆一下RabbitMQ死信队列的基本概念。根据引用中的内容,死信队列是当消息无法被正常消费时,会被转发到另一个队列进行处理,比如处理超时未支付的订单。这点用户提到的引用1和5都有说明,所以需要确保在回答中提到这一点。 接下来,用户需要具体的Java实现步骤。引用4提到了原队列和死信队列的声明步骤,还有完整示例。可能需要分步骤讲解,比如配置原队列、死信交换机、绑定等。同时,引用3提到了延迟消息插件的使用,但用户的问题可能更关注死信队列本身,而不是延迟队列,所以暂时可能不需要涉及插件,除非用户需要处理延迟的情况。不过,引用1中的例子是用死信队列处理超时订单,所以可能需要设置消息的TTL,即生存时间,当消息过期后进入死信队列。 然后,需要确保代码示例的正确性。使用Spring AMQP的话,配置队列时需要设置死信交换机和路由键,这部分在引用4和5中有提到。例如,原队列需要设置x-dead-letter-exchange参数指向死信交换机,并可能设置x-dead-letter-routing-key。 接下来,步骤应该是:首先声明死信交换机和队列,然后绑定它们。接着声明原队列,并设置死信参数。然后编写生产者和消费者代码,生产者发送消息到原队列,如果消息过期或无法处理,会被路由到死信队列,消费者监听死信队列进行处理。 需要注意的细节是,消息成为死信的条件有三个:被拒绝、过期、队列达到最大长度。用户可能需要处理的是过期的情况,比如订单超时,所以示例中需要设置消息的TTL。可以通过队列级别设置,或者单条消息设置。在Java中,可以使用MessagePostProcessor来设置单条消息的TTL。 然后,编写代码示例。使用Spring Boot的话,配置类中声明交换机和队列,设置参数。生产者发送消息时设置过期时间,消费者监听死信队列。需要确保代码中的参数正确,比如交换机的名称、路由键等。 另外,引用2提到RabbitMQ的命令行工具,可以建议用户在测试时使用这些命令查看队列情况,比如rabbitmqctl list_queues,这样方便调试。 可能的问题点:用户可能混淆死信队列延迟队列。根据引用3,延迟队列需要插件支持,而死信队列是原生功能。所以需要明确区分,示例中不需要插件,除非需要更复杂的延迟处理。但用户的问题可能只需要基本的死信队列实现,处理消息过期的情况。 最后,确保回答结构清晰,分步骤说明,并提供完整的代码示例,同时引用相关的内容作为支持。例如,原队列配置参数引用自引用4和5,处理超时订单的场景引用自引用1。此外,生成相关问题,可能包括如何设置不同的死信条件、如何处理死信消息、或者如何与延迟队列结合使用等。</think>### Java实现RabbitMQ死信队列教程 #### 一、实现原理 死信队列(Dead Letter Queue, DLQ)用于接收因过期、被拒绝或队列达到最大长度而无法正常消费的消息[^5]。实现核心是通过定义**原队列**时绑定死信交换机(Dead Letter Exchange, DLX),并设置消息失效后的路由规则[^4]。 #### 二、代码实现步骤(Spring Boot + RabbitMQ) 1. **添加依赖** ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-amqp</artifactId> </dependency> ``` 2. **配置队列与交换机** ```java @Configuration public class RabbitConfig { // 定义原队列(需绑定死信交换机) @Bean public Queue originQueue() { Map<String, Object> args = new HashMap<>(); args.put("x-dead-letter-exchange", "dlx.exchange"); // 死信交换机名称 args.put("x-dead-letter-routing-key", "dlx.routingkey"); // 死信路由键 args.put("x-message-ttl", 10000); // 消息存活时间10秒(测试用) return new Queue("order.queue", true, false, false, args); } // 定义死信交换机和队列 @Bean public DirectExchange dlxExchange() { return new DirectExchange("dlx.exchange"); } @Bean public Queue dlxQueue() { return new Queue("dead.letter.queue"); } @Bean public Binding dlxBinding() { return BindingBuilder.bind(dlxQueue()) .to(dlxExchange()).with("dlx.routingkey"); } } ``` 3. **生产者发送消息** ```java @RestController public class ProducerController { @Autowired private RabbitTemplate rabbitTemplate; @GetMapping("/sendOrder") public String sendOrder() { String message = "订单ID:1001"; rabbitTemplate.convertAndSend( "", // 使用默认交换机 "order.queue", message, msg -> { msg.getMessageProperties().setExpiration("10000"); // 单条消息TTL return msg; }); return "订单已创建,10秒未支付将失效"; } } ``` 4. **消费者监听死信队列** ```java @Component public class DLQConsumer { @RabbitListener(queues = "dead.letter.queue") public void handleDeadLetter(String message) { System.out.println("处理超时订单:" + message); // 执行订单取消逻辑[^1] } } ``` #### 三、验证方法 1. 启动Spring Boot应用 2. 访问`http://localhost:8080/sendOrder`发送订单 3. 等待10秒后查看控制台输出 4. 使用命令查看队列状态: ```bash rabbitmqctl list_queues name messages_ready messages_unacknowledged ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值