RabbitMQ02:队列过期与死信队列

本文介绍了RabbitMQ中队列过期与死信队列的概念和配置方法。过期队列通过设置消息存活时间ttl,当消息未被消费超时后成为死信。设置队列全局ttl或每条消息单独设置ttl,过期消息处理方式不同。同时,死信队列用于存放过期、被拒绝或达到最大长度的消息,通过配置可将死信转发至此队列,进行后续处理。文章提供了相关代码示例进行演示。

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

一、过期队列

消息如果在队列中一直没有被消费且存在时间超过了ttl,消息就会变成死信,后续无法再消费。设置ttl有两种方式:

  • 声明消息队列的时候,这个是全局的,所有发到这个队列的消息的过期时间是一样的
  • 发送消息的时候设置属性,可以每条消息设置不同的ttl
    假如你两种都设置了,以小的ttl为准

区别:queue的全局ttl,消息过期立刻就会被删掉;如果是发送消息时设置的ttl,过期之后并不会立刻删掉,这时候消息是否过期是需要投递给消费者的时候判断的

原因:queue的全局ttl,队列的有效期都一样,先入队列的队列头部,头部也是最早过期的消息,rabbitmq会有一个定时任务从队列的头部开始扫描是否有过期消息即可。而每条设置不同的ttl,只有遍历整个队列才可以筛选出来过期的消息,这样的效率实在是太低,而且如果消息量大了根本不可行,所以rabbitmq在等到消息投递给消费者的时候判断当前消息是否过期,虽然删除的不及时但是不影响功能

注意:注意,ttl队列一般需要设置监听者,因为过期之后我们会有一些通用处理逻辑比如转发到死信队列

  • 代码演示
    1、模拟生产者下单
@Service
public class OrderService {

    @Autowired
    private RabbitTemplate rabbitTemplate;

    //ttl过期模式模拟下单
    public void makeOrderTtl(String userId, String productId, int num) {
        //1、生成订单
        String orderId = UUID.randomUUID().toString();
        System.out.println("过期队列ttl生成订单:" + orderId);
        //2、通过MQ队列完成消息分发
        String exchangeName = "ttl-direct-exchange";
        String routingKey = "ttl";
        rabbitTemplate.convertAndSend(exchangeName, routingKey, orderId);
    }
}

2、绑定过期队列与交换机

@Configuration
public class TTLRabbitmqConfig {

    //1、声明注册direct式交换机
    @Bean
    public DirectExchange ttldirectExchange() {
        return new DirectExchange("ttl-direct-exchange", true, false);
    }

    //2、设置队列的过期时间
    @Bean
    public Queue ttlDirectQueue() {
        HashMap<String, Object> ttlTime = new HashMap<>();
        ttlTime.put("x-message-ttl", 5000);//时间单位为毫秒,必须为整型
        return new Queue("ttl-direct-queue", true, false, false, ttlTime);
    }

    //3、完成交换机和队列的绑定关系
    @Bean
    public Binding ttlDirectBinding() {
        return BindingBuilder.bind(ttlDirectQueue()).to(ttldirectExchange()).with("ttl");
    }
}

3、测试代码


@SpringBootTest
class RabbitmqProducerApplicationTests {

    @Autowired
    private TTLOrderService ttlOrderService;

    @Test
    void contextLoadsTtl() {
        ttlOrderService.makeOrderTtl("1", "1", 12);
    }

}

运行结果:
已经发送消息,过期前:
在这里插入图片描述
过期后:
在这里插入图片描述

二、死信队列

当一个消息过期、被拒绝、达到了队列的最大长度成为死信队列的时候,就会把这些消息发送到一个队列,这个队列即为死信队列

  • 代码演示
    1、生产者生产一个过期的消息
@Service
public class TTLOrderService {

    @Autowired
    private RabbitTemplate rabbitTemplate;

    //ttl过期模式模拟下单
    public void makeOrderTtl(String userId, String productId, int num) {
        //1、生成订单
        String orderId = UUID.randomUUID().toString();
        System.out.println("过期队列ttl生成订单:" + orderId);
        //2、通过MQ队列完成消息分发
        String exchangeName = "ttl-direct-exchange";
        String routingKey = "ttl";
        rabbitTemplate.convertAndSend(exchangeName, routingKey, orderId);
    }
}

2、过期消息队列与交换机绑定(设置消息的过期时间和过期后将要转发到那个死信队列)

@Configuration
public class TTLRabbitmqConfig {

    //1、声明注册direct式交换机
    @Bean
    public DirectExchange ttldirectExchange() {
        return new DirectExchange("ttl-direct-exchange", true, false);
    }

    //2、设置队列的过期时间
    @Bean
    public Queue ttlDirectQueue() {
        HashMap<String, Object> ttlTime = new HashMap<>();
        ttlTime.put("x-message-ttl", 5000);//时间单位为毫秒,必须为整型
        ttlTime.put("x-dead-letter-exchange", "dead-direct-exchange");//消息过期时,绑定死信队列
        ttlTime.put("x-dead-letter-routing-key", "dead");//绑定路由key,fanout模式不需要绑定
        return new Queue("ttl-direct-queue", true, false, false, ttlTime);
    }


    //3、完成交换机和队列的饿绑定关系
    @Bean
    public Binding ttlDirectBinding() {
        return BindingBuilder.bind(ttlDirectQueue()).to(ttldirectExchange()).with("ttl");
    }


}

3、设置死信队列并且绑定交换机


@Configuration
public class DeadRabbitmqConfig {

    //1、声明死信队列的交换机direct
    @Bean
    public DirectExchange deadExchange() {
        return new DirectExchange("dead-direct-exchange", true, false);
    }


    //2、声明死信队列
    @Bean
    public Queue deadDirectQueue() {
        return new Queue("dead-direct-queue", true);
    }


    //3、完成交换机和队列的绑定关系
    @Bean
    public Binding deadDirectBinding() {
        return BindingBuilder.bind(deadDirectQueue()).to(deadExchange()).with("dead");
    }

}


4、测试

@SpringBootTest
class RabbitmqProducerApplicationTests {

    @Autowired
    private TTLOrderService ttlOrderService;

    @Test
    void contextLoadsTtl() {
        ttlOrderService.makeOrderTtl("1", "1", 12);
    }

}

过期队列过期前,消息在此队列中:
在这里插入图片描述
在过期后,此消息进入死信队列:
在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

微笑AJJD

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值