介绍
DLX,死信交换器,消息在一个队列中变成死信之后,会被重新发送到一个特殊的交换器中,那绑定这个交换器的队列就叫死信队列
原因讲解
造成消息变为死信的几个原因:
- 消息被拒绝了,并且requeue被设置为false
- 消息过期
- 队列达到最大长度了
DLX的存在还是非常有用的,因为它可以在处理异常的情况下,消息不能够被正常消费,而被放入死信队列中的情况,后续可以通过消费这个死信队列中的内容来分析当时为什么会变成死信的原因和异常情况,这样就可以优化系统
代码演示
1、添加依赖
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-amqp</artifactId> </dependency>
2、添加配置信息
spring.application.name=springboot_rabbitmq spring.rabbitmq.host=127.0.0.1 spring.rabbitmq.virtual-host=/ spring.rabbitmq.username=guest spring.rabbitmq.password=guest spring.rabbitmq.port=5672
3、编写mq配置类
@Configuration public class RabbitMqDlxConfig { @Bean public Queue queue() { Map<String, Object> props = new HashMap<>(); // 消息的生存时间 10s props.put("x-message-ttl", 10000); // 设置该队列所关联的死信交换器(当队列消息TTL到期后依然没有消费,则加 入死信队列) props.put("x-dead-letter-exchange", "exchange.go.dlx"); // 设置该队列所关联的死信交换器的routingKey,如果没有特殊指定,使用原 队列的routingKey props.put("x-dead-letter-routing-key", "go.dlx"); Queue queue = new Queue("queue.go", true, false, false, props); return queue; } @Bean public Queue queueDlx() { Queue queue = new Queue("queue.go.dlx", true, false, false); return queue; } @Bean public Exchange exchange() { DirectExchange exchange = new DirectExchange("exchange.go", true, false, null); return exchange; } /** * 死信交换器 * @return */ @Bean public Exchange exchangeDlx() { DirectExchange exchange = new DirectExchange("exchange.go.dlx", true, false, null); return exchange; } @Bean public Binding binding() { return BindingBuilder.bind(queue()).to(exchange()).with("go").noargs(); } /** * 死信交换器绑定死信队列 * @return */ @Bean public Binding bindingDlx() { return BindingBuilder.bind(queueDlx()).to(exchangeDlx()).with("go.dlx").noargs(); } }
4、编写接口
@RestController public class DlxController { @Autowired private AmqpTemplate rabbitTemplate; @RequestMapping("/dlx/send") public String distributeGo() { rabbitTemplate.convertAndSend("exchange.go", "go", "发布一条消息"); return "发送成功"; } @RequestMapping("/read") public String getAccumulatedTask() { String notGo = (String) rabbitTemplate.receiveAndConvert("queue.go.dlx"); return notGo; } }
5、测试接口和观察
①/dlx/send:
发送一条消息,并设置了10s过期时间,请求之后,不操作,等待10s
1、首先,请求之前的效果是这样的
2、然后我们请求一下接口,数据如图:
3、然后我们等待10s,效果是这样的
因为10s过了,消息未被消费,就到了queue.go.dlx这个死信队列中了
4、我们来看一下具体为什么会进入死信队列,是不是因为过期没消费呢?我们点击queue.go.dlx名字,进入详情页
②/read:
去死信队列中消费消息
1、首先,请求前的效果是如下图的,队列中无消息,死信队列有一条过期消息待消费
2、然后请求接口。程序控制台会打印出消息文本:发布一条消息
3、然后我们看一下RabbitMQ的控制台表数据
两个队列都已经被清空了
总结
在使用RabbitMQ结合业务使用队列时,我们应该要考虑指定一个死信交换器,并绑定一个死信队列,这样的话,当消息变成死信的时候,这个消息就会被发送到死信队列上,这样就方便我们去查看消息失败的原因了