【RabbitMQ】死信队列

介绍

DLX,死信交换器,消息在一个队列中变成死信之后,会被重新发送到一个特殊的交换器中,那绑定这个交换器的队列就叫死信队列

原因讲解

造成消息变为死信的几个原因:

  1. 消息被拒绝了,并且requeue被设置为false
  2. 消息过期
  3. 队列达到最大长度了

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结合业务使用队列时,我们应该要考虑指定一个死信交换器,并绑定一个死信队列,这样的话,当消息变成死信的时候,这个消息就会被发送到死信队列上,这样就方便我们去查看消息失败的原因

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值