Spring boot 整合 rabbit MQ 死信队列的应用-订单过期自动取消_rabbitmq 死信队列 处理订单过期未支付-程序员宅基地

技术标签: spring boot  死信队列  订单过期自动取消  rabbit mq  

用户下了订单之后,还未付款,在规定的期限内如果没有支付,则这个订单应该标记为取消。

如果实现过期自动取消,下面有几种解决方案

  1. 定时轮询订单,超过期限且未支付
  2. 创建订单后,开启一个消息队列,等待X时间后执行
  3. 通过死信队列回调

其实方案二和死信队列的原理差不多,但是MQ已经拥有类似的机制,所以我们直接沿用即可。

死信队列介绍

  • 死信队列:DLX,dead-letter-exchange
  • 利用DLX,当消息在一个队列中变成死信 (dead message) 之后,它能被重新publish到另一个Exchange,这个Exchange就是DLX

消息变成死信有以下几种情况

  • 消息被拒绝(basic.reject/ basic.nack)并且不再重新投递 requeue=false
  • 消息过期 (rabbitmq Time-To-Live -> messageProperties.setExpiration())
  • 队列超出最大长度

整体的一个流程思路:

创建一个普通的队列(加入死信队列的一些属性),这个消息永远没有人消费,没人消费则过期,过期则通过之前的设置转发回调给另外一个队列

下面是基于Spring boot 的实际用例

  1. 引入依赖
    1.  <dependency>
                  <groupId>org.springframework.boot</groupId>
                  <artifactId>spring-boot-starter-amqp</artifactId>
       </dependency>

       

  2. 配置类
    1.   /**
           * 死信队列交换机标识符  属性值不能改,写死
           */
          private static final String DEAD_LETTER_QUEUE_KEY = "x-dead-letter-exchange";
          /**
           * 死信队列交换机绑定键 标识符  属性值不能改,写死
           */
          private static final String  DEAD_LETTER_ROUTING_KEY = "x-dead-letter-routing-key";
      
      
          /**
           * deadLetterExchange(direct类型交换机)
           *
           * @return
           */
          @Bean("deadLetterExchange")
          public Exchange deadLetterExchange() {
              return ExchangeBuilder.directExchange("DEAD_LETTER_EXCHANGE").durable(true).build();
          }
      
          /**
           * 声明一个死信队列
           * x-dead-letter-exchange   对应  死信交换机
           * x-dead-letter-routing-key  对应 死信队列
           */
          @Bean("deadLetterQueue")
          public Queue deadLetterQueue() {
              //应该像个普通队列,里面多设置了两个参数,这个队列没有被消费或者超时 则通过x-dead-letter-exchange 指明重新回到死信交换机 TEST_SIGN_EXCHANGE
              //交换机
              // 参数
              Map<String, Object> args = new HashMap<>(2);
              // 出现dead letter之后将dead letter重新发送到指定exchange
              args.put(DEAD_LETTER_QUEUE_KEY, "DEAD_LETTER_EXCHANGE");
              // 出现dead letter之后将dead letter重新按照指定的routing-key发送
              args.put(DEAD_LETTER_ROUTING_KEY, "REDIRECT_KEY");
              // name队列名字  durable是否持久化,true保证消息的不丢失, exclusive是否排他队列,如果一个队列被声明为排他队列,该队列仅对首次申明它的连接可见,并在连接断开时自动删除, autoDelete如果该队列没有任何订阅的消费者的话,该队列是否会被自动删除, arguments参数map
              return new Queue("DEAD_LETTER_QUEUE", true, false, false, args);
          }
      
      
          /**
           * 死信路由通过 DEAD_LETTER_KEY 绑定到死信队列上.
           */
          @Bean
          public Binding deadLetterBinding() {
              return new Binding("DEAD_LETTER_QUEUE", Binding.DestinationType.QUEUE, "DEAD_LETTER_EXCHANGE", "DEAD_LETTER_KEY", null);
      
          }
      
          /**
           * 死信路由通过 REDIRECT_KEY 绑定到转发队列上.   这个队列绑定的是当出现死信消息后 重新转发给的队列
           */
          @Bean
          public Binding redirectBinding() {
              return new Binding("REDIRECT_QUEUE", Binding.DestinationType.QUEUE, "DEAD_LETTER_EXCHANGE", "REDIRECT_KEY", null);
          }
          /**
           * 定义死信队列转发队列.   (和普通队列一样,这个队列是为了原有的消息没有被消费重新转发给一个新的队列)
           */
          @Bean("redirectQueue")
          public Queue redirectQueue() {
              return new Queue("REDIRECT_QUEUE", true, false, false);
          }
      

       

  3. 监听类
    1. @Component
      public class TestMQConsumer {
      
           /**
           * 监听转发队列  死信队列重新转发回这里
           *
           */
          @RabbitListener(queues = {"REDIRECT_QUEUE"})
          public void redirect(HashMap<String,Object> dataMap) throws IOException {
              System.out.println(dataMap.get("msg"));
              System.out.println("我是转发队列,这里执行逻辑业务");
          }
      }
      

       

  4. 执行类
    1.   @Test
          public void testMq(){
              //声明消息处理器 设置消息的编码以及消息的过期时间 时间毫秒值为字符串
              MessagePostProcessor messagePostProcessor = message -> {
                  MessageProperties messageProperties = message.getMessageProperties();
                  messageProperties.setMessageId(UUID.randomUUID().toString().replaceAll("-", ""));
                  messageProperties.setContentEncoding("utf-8");
                  //超时时间10秒
                  messageProperties.setExpiration(String.valueOf(1000*10));
                  return message;
              };
              Map<String, Object> dataMap = new HashMap<>();
              dataMap.put("msg","我是传递的消息");
              rabbitTemplate.convertAndSend("DEAD_LETTER_EXCHANGE", "DEAD_LETTER_KEY",dataMap,messagePostProcessor);
          }

       

执行效果:

DEAD_LETTER_QUEUE 执行后没有被消费,超过10秒钟后自动回调 REDIRECT_QUEUE 

继续学习呀!!

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/p_moriarty/article/details/102218596

智能推荐

视频教程-跟一夫学UI设计 APPUI综合设计与图标实战案例视频教程 photoshop绘制icon案例-UI-程序员宅基地

文章浏览阅读104次。跟一夫学UI设计 APPUI综合设计与图标实战案例视频教程 photoshop绘制icon案例 ..._app风格案例视频

vue 海康视频播放_vue-hkvideo-程序员宅基地

文章浏览阅读4k次,点赞4次,收藏22次。1. 下载并安装海康 web 插件https://open.hikvision.com/download/5c67f1e2f05948198c909700?type=102. 把上一步解压的三个 js, 复制到你的项目中, 根据路径, 自己引入到 index.html 中3. 建议运行它的 demo, 大概看看代码, 了解一下它的大致结构, 它的注解很详细, 3 分钟就能看完4. 贴上我的代码(我的是每次只显示一个画面, 点击摄像头切换画面)<temp.._vue-hkvideo

html li 鼠标经过变色,CSS实现li标签鼠标经过时改变背景颜色-程序员宅基地

文章浏览阅读5.3k次,点赞3次,收藏4次。很多时候需要用到这个css效果,实际上就用了一个li标签的热点样式,不仅是li标签,div等也可以的完整代码如下,div/css鼠标热点改变li标签背景颜色body{ background-color:#CCCC99; margin:0; padding:0; color:#fff;}ul{ margin:0; padding:50px;}li{ list-style:none; height:2..._ul li 样式 鼠标移入颜色

数据恢复:在 Linux 上恢复删除了的文件_linux系统,删了某一个文件夹的数据还清空了回收站,还能不能找回来我的数据-程序员宅基地

文章浏览阅读238次,点赞4次,收藏8次。把删除创建为rm -i 的别名当 -i 选项配合 rm 命令(也包括其他文件处理命令比如 cp 或者 mv)使用时,在删除文件前会出现一个提示。其中,/home/gacanepa/rescued 是另外一个磁盘中的目录 - 请记住,把文件恢复到被删除文件所在的磁盘中不是一个明智的做法。安装完成后,我们做一个简单的测试吧。如果在恢复过程中,占用了被删除文件之前所在的磁盘分区,就可能无法恢复文件。但愿你对于你的文件足够小心,当你要从外部磁盘或 USB 设备中恢复丢失的文件时,你只需使用这个工具即可。

2021-09-15 WPF上位机 15-属性绑定(数据格式化)_wpf 自定义属性绑定 格式化 实现-程序员宅基地

文章浏览阅读3.2w次。<Window x:Class="Zhaoxi.BindingStudy.DataFormatStudy.DataFormatStudyWin" xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:d="http://schemas.microsoft.._wpf 自定义属性绑定 格式化 实现

[常用办公软件] wps怎么自动生成目录?wps自动生成目录的设置教程_wps目录自动生成-程序员宅基地

文章浏览阅读1.1w次,点赞3次,收藏5次。转载请说明来源于"厦门SEO"本文地址:http://www.96096.cc/Article/160880.html常用办公软件  WPS Office是由金山软件股份有限公司开发的一款针对个人永久免费的办公软件,在我们的日常生活和工作中,WPS Office比起微软Microsoft Office来说在文字上的处理会更深入国人用户的人心,熟悉操作WPS的办公小技巧,能够更高效的提高我们的工作效率,今天小编要为大家分享的是WPS怎么自动生成目录?快来一起看看WPS自动生成目录的设置教程吧。_wps目录自动生成

随便推点

RecyclerView实现吸顶效果项目实战(三):布局管理器LayoutManager-程序员宅基地

文章浏览阅读338次,点赞4次,收藏6次。架构师不是天生的,是在项目中磨练起来的,所以,我们学了技术就需要结合项目进行实战训练,那么在Android里面最常用的架构无外乎 MVC,MVP,MVVM,但是这些思想如果和模块化,层次化,组件化混和在一起,那就不是一件那么简单的事了,我们需要一个真正身经百战的架构师才能讲解透彻其中蕴含的深理。此时,RecyclerView第一个item是添加进Adapter中的最后一个,最后一个item是第一个加进Adapter的数据,RecyclerView会自动滑到末尾,另外item整体是依靠下方的。

【智能排班系统】基于AOP和自定义注解实现接口幂等性-程序员宅基地

文章浏览阅读884次。使用多种方式实现接口幂等性,通过定义注解方便对方法进行幂等性控制

SpringBoot整合Swagger2 详解_springboot swagger2 开关-程序员宅基地

文章浏览阅读324次。SpringBoot、Swagger2 整合详解_springboot swagger2 开关

spring boot 项目报错 java.sql.SQLException: The server time zone value '�й���׼ʱ��' is unrecognized_springboot项目里面报错 the server time zone value ' й-程序员宅基地

文章浏览阅读2.8w次,点赞96次,收藏115次。报错说是时区不对因为mysql-connection-java版本导致时区的问题。pom.xml:控制台报错信息:java.sql.SQLException: The server time zone value ‘�й���׼ʱ��’ is unrecognized or represents more than one time zone. You must configure ei..._springboot项目里面报错 the server time zone value ' й

最全Android Kotlin 学习路线(Kotlin 从入门、进阶到实战)_kotlin学习-程序员宅基地

文章浏览阅读4.2k次。Kotlin 是由 jetBrains 开发的一门现代多平台应用的静态编程语言,Kotlin 代码即可以编译成 Java 字节码,又可以编译成 JavaScript,Kotlin 是开源的,源码在这。Kotlin 包含了大量的语法糖,在编码的时候,会大大的简化我们的代码量及工作效率。且相比传统的 Java 语言,Kotlin 种大量的简写,可以减少很多用Java 必须要写的样板代码,减少大量的 if…else 等嵌套,减少大量接口的实现,代码结构也会更加清晰。_kotlin学习

【前端素材】推荐优质新鲜绿色蔬菜商城网站设计Harmic平台模板(附源码)-程序员宅基地

文章浏览阅读753次,点赞30次,收藏21次。在线绿色新鲜果蔬商店网站是指一个专门销售新鲜、绿色、有机水果和蔬菜的电子商务平台。这类网站旨在为消费者提供方便、快捷的购买渠道,同时确保他们能够购买到高质量、新鲜的产品。