RabbitMQ 高级特性——死信队列

news2024/11/29 2:31:02

在这里插入图片描述

文章目录

  • 前言
  • 死信队列
    • 什么是死信
    • 常见面试题
    • 死信队列的概念:
    • 死信的来源(造成死信的原因有哪些)
    • 死信队列的应用场景

前言

前面我们学习了为消息和队列设置 TTL 过期时间,这样可以保证消息的积压,那么对于这些过期了的消息,它们都去了哪里呢?可以直接将这些消息丢弃,也可以将这些消息单独放在一个叫做 “死信队列” 这样的结构中,那么什么是死信队列呢?这篇文章将为大家介绍一下什么是死信队列。

死信队列

什么是死信

死信(Dead Letter,简称DL)指的是那些由于某些原因无法被正常消费的消息。

那么死信队列就是指存储死信的队列。当消息在一个队列中变成死信之后,它能被重新发送到另一个交换机中,这个交换机就是 DLX(Dead Letter Exchange),绑定 DLX 的队列,就称为死信队列 DLQ(Dead Letter Queue)。

在这里插入图片描述

什么样的消息最终才会变成死信呢?

  1. 消息被拒绝,并且设置这个被拒绝的消息不重新进入队列
  2. 消息过期
  3. 队列达到最大长度

如何声明死信交换机和死信队列

包含两个部分:

  • 声明正常的交换机和队列
  • 声明死信交换机和死信队列
public static final String NORMAL_EXCHANGE = "normal.exchange";
public static final String NORMAL_QUEUE = "normal.queue";
public static final String DL_EXCHANGE = "dl.exchange";
public static final String DL_QUEUE = "dl.queue";

声明交换机、队列以及交换机和队列的绑定关系:

@Configuration
public class DLconfig {
    @Bean("normalExchange")
    public DirectExchange normalExchange() {
        return ExchangeBuilder.directExchange(Constants.NORMAL_EXCHANGE).durable(true).build();
    }

    //将正常队列和死信交换机进行绑定,这是第一这种绑定队列和死信交换机的方法
    @Bean("normalQueue")
    public Queue normalQueue() {
        return QueueBuilder.durable(Constants.NORMAL_QUEUE)
                .deadLetterExchange(Constants.DL_EXCHANGE)
                .deadLetterRoutingKey("dl")
                .build();
    }

	//这是正常队列和死信交换机绑定的第二种方式,将配置信息以键值对的方式传递
	@Bean("normalQueue1")
    public Queue normalQueue1() {
        Map<String,Object> arguments = new HashMap<>();
        arguments.put("x-dead-letter-exchange",Constants.DL_EXCHANGE); //绑定死信交换机
        arguments.put("x-dead-letter-routing-key","dl"); //设置发送给死信交换机的routing key,死信交换机会根据这个routing key将死信传递给指定的死信队列
        return QueueBuilder.durable(Constants.NORMAL_EXCHANGE).withArguments(arguments).build();
    }

    @Bean("normalBinding")
    public Binding normalBinding(@Qualifier("normalQueue") Queue queue, @Qualifier("normalExchange") DirectExchange exchange) {
        return BindingBuilder.bind(queue).to(exchange).with("normal");
    }

    @Bean("dlExchange")
    public DirectExchange dlExchange() {
        return ExchangeBuilder.directExchange(Constants.DL_EXCHANGE).durable(true).build();
    }

    @Bean("dlQueue")
    public Queue dlQueue() {
        return QueueBuilder.durable(Constants.DL_QUEUE).build();
    }

	@Bean("dlBinding")
    public Binding dlBinding(@Qualifier("dlQueue") Queue queue,@Qualifier("dlExchange") DirectExchange exchange) {
        return BindingBuilder.bind(queue).to(exchange).with("dl");
    }
}

制造死信条件:

前面说了造成死信的原因有三个:1. 消息过期 2. 消息被消费者拒绝且消息不重新进入队列 3. 队列到达最大长度之后,再进入队列的消息就会成为死信。

那么我们制造出死信,就是让队列中的消息数达到最大的长度:

@Bean("normalQueue")
public Queue normalQueue() {
    return QueueBuilder.durable(Constants.NORMAL_QUEUE)
            .deadLetterExchange(Constants.DL_EXCHANGE)
            .deadLetterRoutingKey("dl")
            .maxLength(10)
            .build();
}

然后我们的生产者一次产生 20 条消息:

@RequestMapping("/dl")
public String dl() {
    for (int i = 0; i < 20; i++) {
        rabbitTemplate.convertAndSend(Constants.NORMAL_EXCHANGE,"normal","rabbitmq dl" + i);
    }
    return "消息发送成功";
}

NORMAL_QUEUE 我们就先不设置消费者,然后为死信队列设置消费者,看看超出队列长度的消息是否成为了死信最终到达了死信队列:

@Component
public class DLListener {
    @RabbitListener(queues = Constants.DL_QUEUE)
    public void listener(Message message, Channel channel) {
        System.out.println("死信消费者接收到消息:" + message + channel);
    }
}

在这里插入图片描述
在这里插入图片描述

常见面试题

死信队列的概念:

RabbitMQ中的死信队列(Dead Letter Queue,简称DLQ)是一种特殊的队列机制,用于处理那些因为某些原因无法被正常消费的消息。这些消息在RabbitMQ中被称为“死信”。死信队列的主要目的是防止消息在系统中永久丢失,同时提供一个机制来对这些无法被消费的消息进行后续处理。

死信的来源(造成死信的原因有哪些)

  1. 消息被拒绝
  • 消费者显式拒绝:消费者可以使用RabbitMQ的basic.reject或basic.nack方法拒绝消息。如果requeue参数被设置为false,则消息不会被重新放回队列,而是会被发送到死信队列(如果配置了的话)。
    消费者超时未处理:在某些情况下,如果消费者在一定时间内未能处理消息(如因为性能问题或长时间未响应),消息也可能被视为被拒绝。
  1. 消息过期
  • 设置TTL(Time To Live):RabbitMQ允许为消息或队列设置生存时间(TTL)。如果消息在设置的TTL时间内未被消费,它将被视为过期消息,并可能被发送到死信队列(取决于队列的配置)。
  1. 队列达到最大长度
  • 队列长度限制:RabbitMQ允许为队列设置最大长度限制(可以是消息数量或消息总字节数)。当队列中的消息数量或总字节数达到此限制时,新的消息可能无法被添加到队列中。如果配置了死信队列,则某些消息(如最老的消息)可能会被丢弃或发送到死信队列。

死信队列的应用场景

  1. 错误消息的重试与恢复
    当消息因为消费者代码错误、外部系统不可用或其他暂时性问题而无法被处理时,可以将这些消息发送到死信队列。然后,可以编写专门的消费者来从死信队列中拉取消息,并根据需要进行重试或人工干预。例如,可以设置重试策略,如延迟重试、增加重试次数限制等,以减少因瞬时故障而导致的消息丢失。

  2. 异常消息的分析与监控
    死信队列中的消息通常是因为某些异常或错误而产生的。通过监控和分析这些消息,可以及时发现系统中的问题,如消费者代码中的逻辑错误、外部系统的不可用状态等。这有助于快速定位问题并采取相应的解决措施,保证系统的稳定性和可靠性。

  3. 消息过期处理
    对于设置了TTL(Time To Live)的消息,如果它们在指定的时间内未被消费,将被发送到死信队列。这可以用于处理那些需要在一定时间内得到响应但未能及时处理的消息。例如,在订单系统中,如果某个订单状态更新消息在一定时间内未被处理,可以将其发送到死信队列,并由专门的消费者进行过期处理,如取消订单、发送通知等。

  4. 消息审计与合规性检查
    在某些业务场景中,需要对消息进行审计或合规性检查。通过将无法通过审计或合规性检查的消息发送到死信队列,可以确保只有符合要求的消息被进一步处理。这有助于满足行业监管要求,保护企业利益,并避免潜在的法律风险。

  5. 消息路由与分流
    在某些复杂的消息系统中,可能需要根据消息的不同属性或处理结果将其路由到不同的队列中。通过使用死信队列作为中间环节,可以实现更灵活的消息路由和分流策略。例如,可以将无法被当前消费者处理的消息发送到死信队列,并由另一个消费者进行进一步处理或路由。

  6. 消息备份与恢复
    在某些情况下,可能需要将重要消息进行备份以防止数据丢失。通过将消息发送到死信队列(作为备份队列),可以在系统发生故障或数据丢失时进行恢复。这有助于确保数据的安全性和完整性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2206328.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【更新】上市公司企业机构投资者实地调研数据(2013-2023年)

一、测算方式&#xff1a; 参考《会计研究》逯东&#xff08;2019&#xff09;老师的做法&#xff0c;考虑投资者实地调研的频率和可能性&#xff0c;设立了下述变量来衡量上市公司接待投资者调研情况: 首先&#xff0c;使用年度范围内接待投资者调研的总次数 ( Visitnmb) 作为…

卸载PLSQL及标准卸载流程

目录 1. 卸载PLSQL2. 删除注册表3. 删除数据信息 1. 卸载PLSQL 等待进度条走完 2. 删除注册表 regedit 右击删除 3. 删除数据信息 由于AppData是隐藏文件&#xff0c;需要勾选隐藏的项目。 重启电脑&#xff0c;PLSQL就卸载成功了。

低代码工单管理app评测,功能与效率解析

预计到2030年&#xff0c;低代码平台市场将达1870亿美元。ZohoCreator助力企业构建定制化软件应用&#xff0c;以建筑行业工作订单管理app为例&#xff0c;简化流程&#xff0c;提升管理效率&#xff0c;降低成本。其用户友好界面、自动化管理、跨平台使用及全面报告功能受企业…

项目优化内容及实战

文章目录 事前思考Prometheus 普罗米修斯概述架构安装及使用 Grafana可视化数据库读写分离实战1-PrometheusGrafanaspringboot 事前思考 需要了解清楚&#xff1a;需要从哪些角度去分析实现&#xff1f;使用了缓存&#xff0c;就需要把缓存命中率数据进行收集&#xff1b;使用…

企业在隔离网环境下如何进行安全又稳定的跨网文件交换?

在数字化时代&#xff0c;企业的数据流通如同血液一般重要。然而&#xff0c;当企业内部实施了隔离网环境&#xff0c;跨网文件交换就成了一个棘手的问题。今天我们将探讨在隔离网环境下&#xff0c;企业面临的跨网文件交换挑战&#xff0c;以及如何通过合规的跨网文件交换系统…

数字电路——触发器1(RS和钟控触发器)

触发器&#xff1a;能够存储一位二进制信息的基本单元电路称触发器(Flip-Flop) 特点&#xff1a; 具有两个能自行保持的稳定状态&#xff0c;用来表示逻辑状态的“0”或“1”。具有一对互补输出。有一组控制(激励、驱动)输入。或许有定时(时钟)端CP(Clock Pulse)。在输入信号…

PostgreSQL 16.4安装以及集群部署

1. 环境准备 1.1 主机环境 主机 IP: 192.24.215.121操作系统: CentOS 9PostgreSQL 版本: 16.4 1.2 从机环境 从机 IP: 192.24.215.122操作系统: CentOS 9PostgreSQL 版本: 16.4 2. 安装 PostgreSQL 16.4 在主从两台机器上都需要安装 PostgreSQL 16.4。 2.1 添加 Postgre…

银行卡基础信息查询 API 对接说明

本文将介绍一种 银行卡基础信息查询 API 对接说明&#xff0c;它可用于银行卡基础信息查询。 接下来介绍下 银行卡基础信息查询 API 的对接说明。 申请流程 要使用 API&#xff0c;需要先到 银行卡基础信息查询 API 对应页面申请对应的服务&#xff0c;进入页面之后&#xf…

Python自定义异常类:实际应用示例之最佳实践

Python自定义异常类&#xff1a;实际应用示例之最佳实践 前言 在软件开发中&#xff0c;合理处理异常是保证程序稳定性的重要环节。虽然 Python 内置了丰富的异常类型&#xff0c;但在处理复杂业务逻辑时&#xff0c;自定义异常类能够使代码更加清晰且具备可扩展性。 本文将…

一个架构师的职业素养:四种常用的权限模型

你好,我是看山。 本文收录在《一个架构师的职业素养》专栏。日拱一卒,功不唐捐。 今天咱们一起聊聊权限系统。 以大家熟知的电商场景举例: 用户可以分为普通用户、VIP用户:我们需要控制不同角色用户的访问范围。比如,京东的PLUS会员,可以进入会员专区,而且能够使用礼金…

ESP32接入扣子(Coze) API使用自定义智能体

使用ESP32接入Coze API实现聊天机器人的教程 本示例将使用ESP32开发板通过WiFi接入 Coze API&#xff0c;实现一个简单的聊天机器人功能。用户可以通过串口向机器人输入问题&#xff0c;ESP32将通过Coze API与智能体进行通信&#xff0c;并返回对应的回复。本文将详细介绍了如…

OpenGL 进阶系列03 - OpenGL实例化渲染来提高性能

目录 一:概述 二:实例化渲染的优点: 三:OpenGL实例化渲染的例子: 一:概述 OpenGL 实例化渲染(Instanced Rendering)是一种渲染技术,可以有效地绘制多个相同对象,而不需要为每个对象单独提交绘制调用。通过这种方式,可以显著提高渲染性能,尤其是在需要绘制大量相…

【每日刷题】Day137

【每日刷题】Day137 &#x1f955;个人主页&#xff1a;开敲&#x1f349; &#x1f525;所属专栏&#xff1a;每日刷题&#x1f34d; &#x1f33c;文章目录&#x1f33c; 1. 1576. 替换所有的问号 - 力扣&#xff08;LeetCode&#xff09; 2. 495. 提莫攻击 - 力扣&#xf…

【数据结构与算法】线性表顺序存储结构

文章目录 一.顺序表的存储结构定义1.1定义1.2 图示1.3结构代码*C语言的内存动态分配 二.顺序表基本运算*参数传递2.1建立2.2初始化(InitList(&L))2.3销毁(DestroyList(&L))2.4判断线性表是否为空表(ListEmpty(L))2.5求线性表的长度(ListLength(L))2.6输出线性表(DispLi…

基于GoogleNet深度学习网络的手语识别算法matlab仿真

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 5.算法完整程序工程 1.算法运行效果图预览 (完整程序运行后无水印) 手语How are you&#xff0c;测试识别结果如下&#xff1a; 手语I am fine&#xff0c;测试识别结果如下&#xff1a; 手…

java入门和Java语法

Java直接运行源代码文件&#xff0c;不会产生HelloWorld.class第二种方法&#xff1a;把模块放在D盘下&#xff0c;然后导入 第三种方法&#xff1a;新建一个模块&#xff0c;然后把内容复制过去 byte l 12; short m l; System.out.println(m); char n a; int reason mn; Sy…

消息摘要算法

算法特点 a) 消息摘要算法/单向散列函数/哈希函数 b) 不同长度的输入&#xff0c;产生固定长度的输出 c) 散列后的密文不可逆 d) 散列后的结果唯一 e) 哈希碰撞 f) 一般用于校验数据完整性、签名sign 由于密文不可逆&#xff0c;所以服务端也无法解密 想要验证&#xf…

解锁机器人视觉与人工智能的潜力,从“盲人机器”改造成有视觉能力的机器人(下)

机器视觉产业链全景回顾 视觉引导机器人生态系统或产业链分为三个层次。 上游&#xff08;供应商&#xff09; 该机器人视觉系统的上游包括使其得以运行的硬件和软件提供商。硬件提供商提供工业相机、图像采集卡、图像处理器、光源设备&#xff08;LED&#xff09;、镜头、光…

如何让信息学奥赛学习“边玩边学”?——趣味编程让枯燥学习变得有趣

信息学奥赛&#xff08;NOI&#xff09;作为一项高水平的编程竞赛&#xff0c;内容涉及到大量的算法、数据结构和复杂的逻辑思维&#xff0c;对学生的要求非常高。然而&#xff0c;面对枯燥的知识点和高难度的题目&#xff0c;很多学生在备赛过程中容易感到乏味甚至放弃。那么&…

Overfrp内网穿透:使用域名将内网http/https服务暴露到公网

项目地址&#xff1a;https://github.com/sometiny/overfrp 使用overfrp部署穿透服务器&#xff0c;绑定域名后&#xff0c;可使用域名访问内网的http/https服务。 用例中穿透服务器和内网机器之间的访问全链路加密&#xff0c;具有ssh2相当的安全级别。&#xff01;&#xf…