rabbitmq系列(四)死信队列
一、什么是死信队列
当消息在一个队列中变成一个死信之后,它将被重新publish到另一个交换机上,这个交换机我们就叫做死信交换机,私信交换机将死信投递到一个队列上就是死信队列。具体原理如下图:

消息变成死信的三种情况:
- 消息被拒绝(basic.reject / basic.nack),并且requeue = false
- 消息TTL过期
- 队列达到最大长度
二、手动签收应答模式
应答模式分为两种,手动签收和自动签收,自动应答就是消费者消费了一条消息就自动告诉队列删除消息。这样的弊端就是不管消费逻辑有没有成功,都会将消息删除,这样就会造成消息丢失。而使用手动签收后,就是在消费逻辑处理成功后,手动告诉队列消费成功,然后队列再去删除这条消息。
- 再消费者配置文件中开启手动签收模式
spring.rabbitmq.listener.simple.acknowledge-mode = manual
- 在消费逻辑处理成功后手动签收,修改消费者代码
@RabbitListener(queues = QUEUE_NAME)
public void receiveMessage(Message message,@Headers Map<String,Object> headers, Channel channel) throws Exception {
Jedis jedis = new Jedis("localhost", 6379);
String messageId = message.getMessageProperties().getMessageId();
String msg = new String(message.getBody(),"UTF-8");
System.out.println("接收导的消息为:"+msg+"==消息id为:"+messageId);
String messageIdRedis = jedis.get("messageId");
if(messageId == messageIdRedis){
return;
}
JSONObject jsonObject = JSONObject.parseObject(msg);
String email = jsonObject.getString("email");
String content = jsonObject.getString("timestamp");
String httpUrl = "http://127.0.0.1:8080/email?email"+email+"&content="+content;
// 如果发生异常则返回null
String body = HttpUtils.httpGet(httpUrl, "utf-8");
//
if(body == null){
throw new Exception();
}
jedis.set("messageId",messageId);
Long deliveryTag = (Long)headers.get(AmqpHeaders.DELIVERY_TAG);
// 手动签收
channel.basicAck(deliveryTag,false);
}
三、产生死信的三种情况演示
- 消息被拒绝
我们继续修改一下消费者代码,尝试让消费者消费的时候发生异常。然后在catch块中拒绝消息。
// 拒绝消息,给死信队列
channel.basicNack(message.getMessageProperties().getDeliveryTag(),false,false);
我们运行程序后发现,**当消息消费异常后在队列”zb-byte1“中的消息被消费了,同时发现在死信队列”dead-byte-zb“中有一条未被消费的消息。**消息到死信队列后,然后我们在创建一个消费者去消费消息就可以了。当然死信队列也需要去手动签收消息。
- 消息TTL过期
这种模式我们也叫做延迟消费,有一种特别经典的案例就是用户在一个商品抢购系统中,用户抢到商品后需要在30分钟时间内支付,不然订单无效。这时候我们就可以通过消息TTL过期来实现,设置队列消息过期时间为30分钟,30分钟后publish到死信队列,我们在死信队列中消费订单状态是否支付成功来判断该订单是否有效。
非常简单,我们只需要在配置死信交换机的时候设置有效时间就可以了
@Bean
public Queue queue(){
Map<String,Object> map = new HashMap<>();
map.put("x-dead-letter-exchange",BEI_EXCHANGE_NAME);
map.put("x-dead-letter-routing-key",BEI_ROUTING_KEY);
map.put("x-message-ttl",7200); // 队列过期时间
Queue queue = new Queue(QUEUE_NAME,true,false,false,map);
return queue;
}
- 队列达到最大长度
设置队列长度即可:
@Bean
public Queue queue(){
Map<String,Object> map = new HashMap<>();
map.put("x-dead-letter-exchange",BEI_EXCHANGE_NAME);
map.put("x-dead-letter-routing-key",BEI_ROUTING_KEY);
map.put("x-max-length",3);
// map.put("x-message-ttl",7200); // 队列过期时间
Queue queue = new Queue(QUEUE_NAME,true,false,false,map);
return queue;
}
设置好之后,我们先不要启动消费者,然后调用生成者往队列中发送消息,当消息长度大于3时,我们发现消息进入了死信队列。
注意:前文中也提到过,队列不能被修改,也就是说已经创建好的队列设置了过期时常为7200s,然后我们注释掉,增加队列长度是3的代码,这样运行会报错,必须在rabbitmq中将该队列删除,然后重新生成队列才可以。
如果文章对您有帮助,请记得点赞关注哟~
欢迎大家关注我的公众号:字节传说,每日推送技术文章供大家学习参考。
rabbitmq系列(四)死信队列的更多相关文章
- 关于 RabbitMQ 的 Dead-Letters-Queue “死信队列”
来自一个队列的消息可以被当做‘死信’,即被重新发布到另外一个“exchange”去,这样的情况有: 消息被拒绝 (basic.reject or basic.nack) 且带 requeue=fa ...
- Rabbitmq消费失败死信队列
Rabbitmq 重消费处理 一 处理流程图: 业务交换机:正常接收发送者,发送过来的消息,交换机类型topic AE交换机: 当业务交换机无法根据指定的routingkey去路由到队列的时候,会全部 ...
- RabbitMQ TTL、死信队列
TTL概念 TTL是Time To Live的缩写,也就是生存时间. RabbitMQ支持消息的过期时间,在消息发送时可以进行指定. RabbitMQ支持队列的过期时间,从消息入队列开始计算,只要超过 ...
- RabbitMQ (五):死信队列
什么是TTL RabbitMQ的TTL全称为Time-To-Live,表示的是消息的有效期.消息如果在队列中一直没有被消费并且存在时间超过了TTL,消息就会变成了"死信" (Dea ...
- rabbitmq系列四 之路由
1.路由 在上一个的教程中,我们构建了一个简单的日志记录系统.我们能够向许多接收者广播日志消息. 在本次教程中,我们向该系统添加一些特性,比如,我只需要严重错误(erroe级别)的部分日志打印到磁盘文 ...
- RabbitMQ系列(四)--消息如何保证可靠性传输以及幂等性
一.消息如何保证可靠性传输 1.1.可能出现消息丢失的情况 1.Producer在把Message发送Broker的过程中,因为网络问题等发生丢失,或者Message到了Broker,但是出了问题,没 ...
- RabbitMQ使用 prefetch_count优化队列的消费,使用死信队列和延迟队列实现消息的定时重试,golang版本
RabbitMQ 的优化 channel prefetch Count 死信队列 什么是死信队列 使用场景 代码实现 延迟队列 什么是延迟队列 使用场景 实现延迟队列的方式 Queue TTL Mes ...
- RabbitMQ死信队列
关于RabbitMQ死信队列 死信队列 听上去像 消息“死”了 其实也有点这个意思,死信队列 是 当消息在一个队列 因为下列原因: 消息被拒绝(basic.reject/ basic.nac ...
- 【MQ中间件】RabbitMQ -- RabbitMQ死信队列及内存监控(4)
1.RabbitMQ TTL及死信队列 1.1.TTL概述 过期时间TTL表示可以对消息设置预期的时间,在这个时间内都可以被消费者接收获取:过了之后消息将自动被删除.RabbitMQ可以对消息和队列设 ...
随机推荐
- 关于JavaScript中的typeof与instanceof
JavaScript中typeof和instanceof可以用来判断一个数据的类型,什么时候选择使用typeof?什么时候选择使用instanceof? typeof运算符 typeof运算符返回值有 ...
- python djangjo完整的实现添加的实例
实现:点击添加实现模态对话框,添加数据并显示. urls.py from django.conf.urls import url from django.contrib import admin fr ...
- [洛谷P3366] [模板] 最小生成树
存个模板,顺便复习一下kruskal和prim. 题目传送门 kruskal 稀疏图上表现更优. 设点数为n,边数为m. 复杂度:O(mlogm). 先对所有边按照边权排序,初始化并查集的信息. 然后 ...
- c中结构体边界对齐
原则1.普通数据成员对齐规则:第一个数据成员放在offset为0的地方,以后每个数据成员存储的起始位置要从该成员大小的整数倍开始(比如int在32位机为4字节,则要从4的整数倍地址开始存储). 原则2 ...
- Java JDBC调用inout类型参数的存储过程
存储过程参数类型:in.out.inout,in:输入类型,out:输出类型,inout:既可输入,也可以输出. 一.JDBC调用inout类型参数的存储过程,并且获得返回值 Class.forNam ...
- 关于sql拼接的知识点 where1=1
String sql="select * from tab_route where 1 = 1 "; 这样不会报错,而且可以根据情况,再去拼接sql 可以使用if(){}else{ ...
- Bar条形图
import matplotlib.pyplot as plt import numpy as np n = 12 X = np.arange(n) Y1 = (1 - X / float(n)) * ...
- Binder机制简析(三)
注册Service Service组件运行在Server进程中,首先要将Service注册到Service Manager中,再启动一个Binder线程池来等待和处理Client的通信请求. 注册过程 ...
- zctf 2016 android writeup - Jieming的博客
本文为2016年zctf中android的writeup. 首先点我下载题目.使用jeb反编译,对username和password进行部分验证后,再将username+password及一个数据库查 ...
- CC03 iOS推送机制浅析
• ios推送机制 可以通俗的把APNS理解为iOS系统为每个app提供的长连接通道 苹果限制了每个app在后台存活的时间,最重要的目的是为了省电,其次优化内存这些.如果彻彻底底的将app杀死了,服务 ...