RabbitMQ死信交换机

目录

1.死信交换机介绍

 2.TTL

3.延迟队列

4.消息堆积问题

5.惰性队列

 6.代码实战


1.死信交换机介绍

当一个队列中信息满足下列情况之一时,可以成为死信(dead letter)

  (1)消费者使用basic.reject(RejectAndDontRequeueRecoverer处理策略)或bastic.nack(ImmediateRequeueMessageRecoverer处理策略)消费失败,requeue(重新入队)设置参数为false;

  (2)消息是一个过期消息,过期无人消费;

  (3)要投递的队列消息堆积满了,最早的消息成为死信;

如果该队列配置了dead-letter-exchange属性,指定交换机,队列设置dead-letter-routing-key属性,设置死信交换机与死信队列的RoutingKey,那么队列中的死信就会投递到这个交换机中,而这个交换机成为死信交换机(Dead Letter Exchange,简称DLX)。

 

 2.TTL

        TTL,也就是Time-To-Live。如果队列中的消息TTL结束仍未消费则会成为死信,TTL超时分为两种情况:

  (1)消息所在的队列设置了存活时间;

  (2)消息本身设置了存活时间;

 

3.延迟队列

利用ttl结合死信交换机,实现了消息发出后,消费者延迟收到消息的结果,这种消息队列成为延迟队列(Delay Queue)模式。

延迟队列使用的场景:

  (1)用户下单后超过15分钟未支付取消支付;

  (2)预约20分钟会议,到时间自动通知参会人员;

订单的超时处理:

生产者生产一条1分钟后超时的订单消息到正常交换机exchange中,但一分钟后仍未消费,消息会被投递到死信交换机dlx-exchange中,并发送到私信队列中,死信队列dlx-queue的消费者拿到消息后,根据消息去查询订单的状态,如果仍然是为支付状态,将订单状态更新为超时状态 

4.消息堆积问题

当生产者发送的消息的速度大于消费者消费的速度,就会导致队列中的消息堆积,直到达到消息的上限,最早接受的消息,可能会被丢弃,成为死信。

解决消费堆积的三种策略:

  (1)增加消费者,增加消费速度

  (2)在消费者内开启线程池加快消息处理速度

  (3)扩大队列的容积

5.惰性队列

从RabbitMq的3.6.0版本开始增加了Lazy Queues的概念,也就是惰性队列。

惰性队列的特征如下:

  (1)接受消息后直接存入磁盘而非内存;

  (2)消费者消费消息才会中磁盘中将消息加载到内存;

  (3)支持数百万的数据存储;

 6.代码实战

先写两个队列和分别自定义一个直连交换机 并且给交换机分别绑定一个队列

    //死信,延迟队列@Beanpublic Queue queueA(){Map<String, Object> config = new HashMap<>();//message在该队列queue的存活时间最大为20秒config.put("x-message-ttl", 20000);//x-dead-letter-exchange参数是设置该队列的死信交换器(DLX)config.put("x-dead-letter-exchange", "ExchangeB"); //x-dead-letter-routing-key参数是给这个DLX指定路由键config.put("x-dead-letter-routing-key", "bb");return new Queue("queueA",true,false,false,config);}@Beanpublic DirectExchange ExchangeA(){return new DirectExchange("ExchangeA");}@Beanpublic Binding bindingA(){return BindingBuilder.bind(queueA()).to(ExchangeA()).with("aa");}@Beanpublic Queue queueB(){return new Queue("queueB");}@Beanpublic DirectExchange ExchangeB(){return new DirectExchange("ExchangeB");}@Beanpublic Binding bindingB(){return BindingBuilder.bind(queueB()).to(ExchangeB()).with("bb");}

 在控制类给交换机发送一条消息

    //死信交换机@RequestMapping("/send5")public String send5(){template.convertAndSend("ExchangeA","aa","4502520");return "😡";}

写一个测试类进行测试

package com.example.consumer;import lombok.extern.slf4j.Slf4j;
import org.springframework.amqp.rabbit.annotation.RabbitHandler;
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.stereotype.Component;@Component
@SuppressWarnings("all")
@Slf4j
@RabbitListener(queues="queueB")
public class ReceiverQB {@RabbitHandlerpublic void process(String id){log.warn("QB接收到:" + id);}
}

 接着启动生产者和消费者,并在RabbitMQ中查看

 

注意:如果在启动消费者时报错,请先在页面上访问,再重新启动消费者 

因为我们设置队列queue的存活时间最大为20秒,所以再隔20秒后消息会自动出来

 

 

确认消息(局部方法处理消息)

默认情况下消息消费者是自动 ack (确认)消息的,如果要手动 ack(确认)则需要修改确认模式为 manual

在消费者的yml文件中进行配置

server:port: 9999
spring:rabbitmq:host: 192.168.169.132password: 123456port: 5672username: springvirtual-host: my_vhostlistener:simple:acknowledge-mode: manual
接着写一个测试类
package com.example.consumer;import com.rabbitmq.client.AMQP;
import com.rabbitmq.client.Channel;
import lombok.extern.slf4j.Slf4j;
import org.springframework.amqp.rabbit.annotation.RabbitHandler;
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.amqp.support.AmqpHeaders;
import org.springframework.boot.autoconfigure.amqp.RabbitProperties;
import org.springframework.messaging.handler.annotation.Header;
import org.springframework.stereotype.Component;@Component
@SuppressWarnings("all")
@Slf4j
@RabbitListener(queues="queueA")
public class ReceiverQA {@RabbitHandlerpublic void process(String id, Channel channel, @Header(AmqpHeaders.DELIVERY_TAG) long tag) throws Exception{log.warn("QA接收到:" + id);channel.basicAck(tag,true);}
}

在页面中刷新上面的方法,可以在idea中看到

如果想要手动否认,拒绝消息

@RabbitHandlerpublic void process(String id, Channel channel, @Header(AmqpHeaders.DELIVERY_TAG) long tag) throws Exception{log.error("QA接收到:" + id);channel.basicReject(tag,true);Thread.sleep(1000);//延迟时间}

因为延迟时间只有1秒钟,而且需要重新进入队列,所以会一直循环,在循环到在我们上面设置的那个20秒钟后会进入QB

也可以拒绝该消息,消息会被丢弃,不会重回队列  

    @RabbitHandlerpublic void process(String id, Channel channel, @Header(AmqpHeaders.DELIVERY_TAG) long tag) throws Exception{log.error("QA接收到:" + id);channel.basicReject(tag,false);Thread.sleep(1000);//延迟时间}

这时候会直接进入QB 

 

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

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

相关文章

React16源码: React中commit阶段的commitRoot的主流程源码实现

commitRoot 1 &#xff09;概述 在react中有 render 阶段 和 commit 阶段&#xff0c;这是两个不同的阶段1 &#xff09;之前的渲染更新都是render阶段在render阶段&#xff0c;会经历一系列的调度&#xff0c;一系列的节点的更新过程需要去重新计算它的 state, props 生成新的…

SimpleDateFormat学习使用

提起SimpleDateFormat类&#xff0c;想必做过Java开发的童鞋都不会感到陌生。没错&#xff0c;它就是Java中提供的日期时间的转化类。这里&#xff0c; 为什么说SimpleDateFormat类有线程安全问题呢&#xff1f;有些小伙伴可能会提出疑问&#xff1a;我们生产环境上一直在使用S…

【mongoDB】集合的创建和删除

目录 1.集合的创建 2. 查看所有集合 3.删除集合 1.集合的创建 格式&#xff1a; db.createCollection ( name ) 例如创建一个名为 bbb 的集合 还可以通过传递一个选项对象来指定集合的属性&#xff0c;例如最大文档的大小&#xff0c;索引选项等 例如 这样创建了一个名为 cc…

Linux CentOS使用Docker搭建laravel项目环境(实践案例详细说明)

一、安装docker # 1、更新系统软件包&#xff1a; sudo yum update# 2、安装Docker依赖包 sudo yum install -y yum-utils device-mapper-persistent-data lvm2# 3、添加Docker的yum源&#xff1a; sudo yum-config-manager --add-repo https://download.docker.com/linux/cen…

如何在IntelliJ IDEA数据库控制台操作Redis

如何在IntelliJ IDEA数据库控制台操作Redis TIPS 本文理论支持IntelliJ IDEA家族所有IDE&#xff08;例如Data Grip等&#xff09;、所有版本理论支持所有基于JDBC的各种GUI工具&#xff01; 最近工作中&#xff0c;经常要操作到Redis&#xff0c;尽管市面上的Redis客户端GUI非…

kotlin 项目中文件显示带.kt 结尾与不带.kt结尾

kotlin 项目中文件显示带.kt 结尾与不带.kt结尾&#xff0c;而且显示的图片也不一样&#xff0c; 首先说下&#xff0c;这个只是开发工具上的一个设计细节展示&#xff0c;无论显示效果是否带.kt 实际的文件都是带.kt结尾的&#xff0c;这个可以到文件的目录下查看文件 创建文…

算法基础之树状数组

文章目录 树状数组 树状数组 树状数组能解决的最关键的问题就是能够 O ( log ⁡ n ) O(\log n) O(logn)内&#xff0c;给某个位置上的数&#xff0c;加上一个数&#xff0c;或者求前缀和 他和前缀和数组的区别就是&#xff0c;树状数组支持修改原数组的内容&#xff0c;而前缀…

代码随想录算法训练营第30天(回溯算法06 | ● 332.重新安排行程 ● 51. N皇后 ● 37. 解数独 ● 总结

回溯算法06 332.重新安排行程&#xff08;可跳过&#xff09;解题思路难点 51.N皇后&#xff08;可跳过&#xff09;解题思路回溯三部曲难点 5. 解数独&#xff08;可跳过&#xff09;解题思路回溯三部曲 总结篇&#xff08;没来及看 332.重新安排行程&#xff08;可跳过&#…

C语言第九弹---二维数组

✨个人主页&#xff1a; 熬夜学编程的小林 &#x1f497;系列专栏&#xff1a; 【C语言详解】 【数据结构详解】 二维数组 1、二维数组的创建 1.1、二维数组的概念 ​1.2、⼆维数组的创建 2、二维数组的初始化 2.1、不完全初始化 ​2.2、完全初始化 ​2.3、按照行初始化 ​2.4、…

如何使用Docker安装Spug并实现远程访问本地运维管理界面

文章目录 前言1. Docker安装Spug2 . 本地访问测试3. Linux 安装cpolar4. 配置Spug公网访问地址5. 公网远程访问Spug管理界面6. 固定Spug公网地址 前言 Spug 面向中小型企业设计的轻量级无 Agent 的自动化运维平台&#xff0c;整合了主机管理、主机批量执行、主机在线终端、文件…

Vue开发之proxy代理的配置(附带uniapp代理配置)

vue 1.在vue.config.js中添加 devServer 属性中配置 proxy 属性 module.exports {productionSourceMap: false,publicPath: /,devServer: {port: 8085,proxy: {/api/admin: {target: http://10.58.104.70:6111,changeOrigin: true,pathRewrite: {/api/: /}},/api: {target: …

自动 卸载或安装 Python第三方库

在调用 pip.exe 时&#xff0c;可以使用相对路径也可以使用绝对路径 路径中如果包含空格&#xff0c;最好使用相对路径&#xff0c;这就要求 pip.exe 所在文件夹设置为环境变量 可以参考&#xff1a; Windows下将文件夹设置为环境变量 echo off setlocal enabledelayedexpansi…

UE创建数据表格

创建一个数据表格需要行结构 继承自FTableRowBase的一个子类 效果 如何使用它 在蓝图中给C该类型的指针变量选用 UDataTable类型的 FindRow()函数可查询并返回对应行的行结构 FTableRowBase GetAllRows()函数可以获得该数据表的所有行、

centos 安装mysql5.7教程

一&#xff0c;配置yum mysql5.7安装源 配置yum mysql5.7安装源 yum localinstall https://dev.mysql.com/get/mysql57-community-release-el7-11.noarch.rpm 配置mysql5.7安装源成功 查看配置成功的安装源 yum repolist enabled | grep "mysql*" 执行后看到已配…

环境监测与预报:探索天气预报查询API在生态保护中的作用

摘要 随着全球气候变化的加剧&#xff0c;生态保护已成为全球关注的焦点。天气预报API作为一种强大的工具&#xff0c;不仅能够提供实时的气象数据&#xff0c;还能在生态保护领域发挥重要作用。本文将探讨天气预报API如何帮助科学家、环保组织和政策制定者更好地理解和预测环…

什么是 Docker

1.什么是 Docker 1.1 官方定义 最新官网首页 # 1.官方介绍 - We have a complete container solution for you - no matter who you are and where you are on your containerization journey. - 翻译: 我们为你提供了一个完整的容器解决方案,不管你是谁,不管你在哪,你都可以…

vue3常用代码

文章目录 监听路由vue3 警告Feature flag __VUE_PROD_HYDRATION_MISMATCH_DETAILS__ is not explicitly defined.mitt、project/inject 无效解决方案 菜鸟做项目时发现很多 vue3 常用的代码&#xff0c;所以来总结一下&#xff01; 监听路由 import { useRoute } from "…

如何禁用 el-table 单独某一行,修改某一行样式等(最有效)

案例&#xff1a;根据el-table :data"tableData"中是否有invalidStatus值为1&#xff0c;如果是就是不禁用&#xff0c;否就禁用这一行&#xff0c;当然这个invalidStatus随意就行&#xff0c;只要在tabledata中的每一行数据中有这个属性就行&#xff0c;也就是row中…

Android创建保存Excel文件

Android开发生成保存Excel文件&#xff0c;首先下载两个jar包。下载地址&#xff1a;Android读写Excel文件的两个jar包资源-CSDN文库 poi-3.12-android-a.jar poi-ooxml-schemas-3.12-20150511-a.jar 把jar包放在app的libs文件夹下&#xff0c;引用jar我一般都在build.gradle的…

代码随想录算法训练营第四十一天(动态规划篇)|理论基础,509. 斐波那契数, 70. 爬楼梯, 746. 使用最小花费爬楼梯

动态规划理论基础 动态规划&#xff1a;每一个状态一定是由上一个状态推导出来的。 贪心&#xff1a;局部直接选最优的 解题步骤 确定dp数组&#xff08;dp table&#xff09;以及下标的含义确定递推公式dp数组如何初始化确定遍历顺序举例推导dp数组 509. 斐波那契数 题目…