死信队列与延迟队列区别与问题优化

本文介绍了RabbitMQ中延时队列和死信队列的区别,强调了死信队列在消息按时间顺序处理上的问题,并提出基于插件的延迟队列解决方案,该方案能确保消息按延迟时间而非发布顺序执行。同时,文章总结了死信队列在其他场景下的应用和优势,并提到了其他技术如DelayQueue、Redis和Kafka等实现延时队列的方法。

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

延时队列

延时队列.队列内部是有序的,最重要的特性就体现在它的延时属性上,延时队列中的元素是希望在指定时间到了以后或之前取出和处理,简单来说, 延时队列就是用来存放需要在指定时间被处理的元素的队列, 延时队列其实就是特殊的死信队列

基于死信队列存在的问题:

如果在消息属性上设置TTL的方式,消息可能并不会按时"死亡",因为RabbitMQ只会检查第一个消息是否过期,如果过期则丢到死信队列,如果第一个消息的延时时长很长, 而第二个消息的延时时长很短,第二个消息并不会优先得到执行,而是在第一个消息过期后才会执行

死信队列应用场景

为保证订单业务消息数据不丢失,需要使用到RabbitMQ的死信队列机制,当消息消费发生异常时,将消息投入死信队列中,还比如说:用户在商城下单成功并点击去支付后在指定时间内未支付时自动失效

特点

由于某些特定的原因导致queue中的某些消息无法被消费

死信的来源

消息TTL过期
队列达到最大长度
消息被拒绝(basic.reject或basic.nack)并且requeue-false

死信的设置
 //正常队列设置死信交换机
        arguments.put("x-dead-letter-exchange",DEAD_EXCHANGE);
        //设置死信Routingkey
        arguments.put("x-dead-letter-routing-key","lisi");
		//设置正常队列的长度限制
        arguments.put("x-max-length",6);
        
   

一般死信的过期时间不在消费者处设置, 而是在生产者处设置,这样不至于把过期时间写死, 便于更灵活的把控时间

死信队列中存在问题的解决方案

基于插件的延迟队列方式可以很好的解决死信队列中每次只会监测第一个消息的过期时间, 基于插件的延迟队列不会受先后顺序影响,而只受其延迟时间影响, 延迟时间短的 , 便会先发送出去
方案的实现流程
与死信队列中需要声明两个队列 两个交换机不同的是 通过基于插件的延迟队列, 只需声明一个普通队列 一个延迟消息类型交换机即可
在这里插入图片描述

上面这个延迟消息类型通过在linux中开启延迟插件后才会显示, 然后就可以在代码中进行交换机的声明定义,
通过一个生产者, 一个延迟消息交换机, 一个普通队列, 一个消费者, 即可实现延迟队列的效果, 且弥补了死信队列中遗留的问题

总结

延时队列在需要延时处理的情景下非常有用,使用RabbitMQ来实现延时队列可以很好的利用RabbitMQ的特性,如:消息可靠发送,消息可靠投递,死信队列来保障消息至少被消费一次以及未被正常处理的消息不会被丢弃,另外 通过RabbitMQ集群的特性, 可以很好的解决单点故障问题, 不会因为某个节点挂掉导致延时队列不可用或者消息丢失, 总的来说,死信队列只是在作为延时队列的时候存在一些小问题,但在其他方面的使用是非常完美的
当然, 延时队列还有很多其他选择, 如利用java的DelayQueue, 利用Redisdzset,利用quartz或者利用kafka的时间轮, 都各有特点, 看需要适用的场景

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值