当前位置: 首页 > news >正文

尚品汇-延迟插件实现订单超时取消(四十五)

目录:

(1)延迟插件封装

(2)基于延迟插件测试

        如何保证消息幂等性?

(3)改造订单service-order模块-实现订单超时取消

(1)延迟插件封装

        

把消息带过去: 

在消息的重试发送消息的方法里封装:retrySendMsg

(2)基于延迟插件测试

service-order模块

 rabbit-util模块配置常量MqConst

/*** 取消订单,发送延迟队列*/
public static final String EXCHANGE_DIRECT_ORDER_CANCEL = "exchange.direct.order.cancel";//"exchange.direct.order.create" test_exchange;
public static final String ROUTING_ORDER_CANCEL = "order.create";
//延迟取消订单队列
public static final String QUEUE_ORDER_CANCEL  = "queue.order.cancel";
//取消订单 延迟时间 单位:秒
public static final int DELAY_TIME  = 10;
rabbit-util模块延迟接口封装:RabbitService
/*** 封装发送延迟消息方法* @param exchange* @param routingKey* @param msg* @param delayTime* @return*/
public Boolean sendDelayMsg(String exchange,String routingKey, Object msg, int delayTime){//  将发送的消息 赋值到 自定义的实体类GmallCorrelationData gmallCorrelationData = new GmallCorrelationData();//  声明一个correlationId的变量String correlationId = UUID.randomUUID().toString().replaceAll("-","");gmallCorrelationData.setId(correlationId);gmallCorrelationData.setExchange(exchange);gmallCorrelationData.setRoutingKey(routingKey);gmallCorrelationData.setMessage(msg);gmallCorrelationData.setDelayTime(delayTime);gmallCorrelationData.setDelay(true);//  将数据存到缓存this.redisTemplate.opsForValue().set(correlationId,JSON.toJSONString(gmallCorrelationData),10,TimeUnit.MINUTES);//  发送消息this.rabbitTemplate.convertAndSend(exchange,routingKey,msg,message -> {//  设置延迟时间message.getMessageProperties().setDelay(delayTime*1000);return message;},gmallCorrelationData);//  默认返回return true;
}
修改retrySendMsg方法 – 添加判断是否属于延迟消息

 MQProducerAckConfig 配置类中修改retrySendMsg方法

//  判断是否属于延迟消息
if (gmallCorrelationData.isDelay()){//  属于延迟消息this.rabbitTemplate.convertAndSend(gmallCorrelationData.getExchange(),gmallCorrelationData.getRoutingKey(),gmallCorrelationData.getMessage(),message -> {//  设置延迟时间message.getMessageProperties().setDelay(gmallCorrelationData.getDelayTime()*1000);return message;},gmallCorrelationData);
}else {//  调用发送消息方法 表示发送普通消息  发送消息的时候,不能调用 new RabbitService().sendMsg() 这个方法this.rabbitTemplate.convertAndSend(gmallCorrelationData.getExchange(),gmallCorrelationData.getRoutingKey(),gmallCorrelationData.getMessage(),gmallCorrelationData);
}

 Contrroller:

利用封装好的工具类 测试发送延迟消息

//  基于延迟插件的延迟消息
@GetMapping("sendDelay")
public Result sendDelay(){//  声明一个时间对象SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");System.out.println("发送时间:"+simpleDateFormat.format(new Date()));this.rabbitService.sendDelayMsg(DelayedMqConfig.exchange_delay,DelayedMqConfig.routing_delay,"iuok",3);return Result.ok();
}

消息没有成功到达队列,会出发回调重新发送,会尝试发送三次,消费者会消费三次 

结果会 回发送三次,也被消费三次!

如何保证消息幂等性?

  1. 使用数据方式
  2. 使用redis setnx 命令解决  --- 推荐

幂等性:执行多次,结果都是一样的

在消费者这里进行实现,消费者不管发送者发送多少条消息,只消费一次

 

@SneakyThrows
@RabbitListener(queues = DelayedMqConfig.queue_delay_1)
public void getMsg2(String msg,Message message,Channel channel){//  使用setnx 命令来解决 msgKey = delay:iuokString msgKey = "delay:"+msg;Boolean result = this.redisTemplate.opsForValue().setIfAbsent(msgKey, "0", 10, TimeUnit.MINUTES);//  result = true : 说明执行成功,redis 里面没有这个key ,第一次创建, 第一次消费。//  result = false : 说明执行失败,redis 里面有这个key//  能: 保证消息被消费成功    第二次消费,可以进来,但是要判断上一个消费者,是否将消息消费了。如果消费了,则直接返回,如果没有消费成功,我消费。//  在设置key 的时候给了一个默认值 0 ,如果消费成功,则将key的值 改为1if (!result){//  获取缓存key对应的数据String status = (String) this.redisTemplate.opsForValue().get(msgKey);if ("1".equals(status)){//  手动确认channel.basicAck(message.getMessageProperties().getDeliveryTag(),false);return;} else {//  说明第一个消费者没有消费成功,所以消费并确认SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");System.out.println("接收时间:"+simpleDateFormat.format(new Date()));System.out.println("接收的消息:"+msg);//  修改redis 中的数据this.redisTemplate.opsForValue().set(msgKey,"1");channel.basicAck(message.getMessageProperties().getDeliveryTag(),false);return;}}SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");System.out.println("接收时间:"+simpleDateFormat.format(new Date()));System.out.println("接收的消息:"+msg);//  修改redis 中的数据this.redisTemplate.opsForValue().set(msgKey,"1");//  手动确认消息channel.basicAck(message.getMessageProperties().getDeliveryTag(),false);
}

发送一次消息 

在发送一次消息:Redis有了就不会消费了

(3)改造订单service-order模块-实现订单超时取消

 ​​​​​​​​​​​​​

 

service-order模块配置队列

添加依赖

<!--rabbitmq消息队列-->
<dependency><groupId>com.atguigu.gmall</groupId><artifactId>rabbit-util</artifactId><version>1.0</version>
</dependency>

 

OrderCanelMqConfig  

package com.atguigu.gmall.order.receiver;@Configuration
public class OrderCanelMqConfig {@Beanpublic Queue delayQueue() {// 第一个参数是创建的queue的名字,第二个参数是是否支持持久化return new Queue(MqConst.QUEUE_ORDER_CANCEL, true);}@Beanpublic CustomExchange delayExchange() {Map<String, Object> args = new HashMap<String, Object>();args.put("x-delayed-type", "direct");return new CustomExchange(MqConst.EXCHANGE_DIRECT_ORDER_CANCEL, "x-delayed-message", true, false, args);}@Beanpublic Binding bindingDelay() {return BindingBuilder.bind(delayQueue()).to(delayExchange()).with(MqConst.ROUTING_ORDER_CANCEL).noargs();}
}

发送消息

创建订单时,发送延迟消息

OrderServiceImpl实现类中:

修改保存订单方法

 

@Override
@Transactional
public Long saveOrderInfo(OrderInfo orderInfo) {.....//发送延迟队列,如果定时未支付,取消订单//交换机、路由key、消息(订单id) 超时时间rabbitService.sendDelayMessage(MqConst.EXCHANGE_DIRECT_ORDER_CANCEL, MqConst.ROUTING_ORDER_CANCEL, orderInfo.getId(), MqConst.DELAY_TIME);// 返回return orderInfo.getId();
}

 

接收消息

传的是订单id,这里orderId会接受到,赋值给他,message不是接受到的消息

package com.atguigu.gmall.order.receiver;@Component
public class OrderReceiver {@Autowiredprivate OrderService orderService;//  监听的消息@SneakyThrows@RabbitListener(queues = MqConst.QUEUE_ORDER_CANCEL)public void cancelOrder(Long orderId , Message message, Channel channel){//  判断当前订单Id 不能为空try {if (orderId!=null){//  发过来的是订单Id,那么你就需要判断一下当前的订单是否已经支付了。//  未支付的情况下:关闭订单//  根据订单Id 查询orderInfo select * from order_info where id = orderId//  利用这个接口IService  实现类ServiceImpl 完成根据订单Id 查询订单信息 ServiceImpl 类底层还是使用的mapperOrderInfo orderInfo = orderService.getById(orderId);//  判断支付状态,进度状态if (orderInfo!=null && "UNPAID".equals(orderInfo.getOrderStatus())&& "UNPAID".equals(orderInfo.getProcessStatus())){//  关闭订单//  int i = 1/0;orderService.execExpiredOrder(orderId);}}} catch (Exception e) {//  消息没有正常被消费者处理: 记录日志后续跟踪处理! e.printStackTrace();}//  手动确认消息 如果不确认,有可能会到消息残留。channel.basicAck(message.getMessageProperties().getDeliveryTag(),false);}
}
package com.atguigu.gmall.model.enums;public enum ProcessStatus {UNPAID("未支付", OrderStatus.UNPAID),PAID("已支付", OrderStatus.PAID),NOTIFIED_WARE("已通知仓储", OrderStatus.PAID),WAITING_DELEVER("待发货", OrderStatus.WAITING_DELEVER),STOCK_EXCEPTION("库存异常", OrderStatus.PAID),DELEVERED("已发货", OrderStatus.DELEVERED),CLOSED("已关闭", OrderStatus.CLOSED),COMMNET("已评价",OrderStatus.FINISHED) ,FINISHED("已完结", OrderStatus.FINISHED) ,PAY_FAIL("支付失败", OrderStatus.UNPAID),SPLIT("订单已拆分", OrderStatus.SPLIT);private String comment ;private OrderStatus orderStatus;ProcessStatus(String comment, OrderStatus orderStatus){this.comment=comment;this.orderStatus=orderStatus;}public String getComment() {return comment;}public void setComment(String comment) {this.comment = comment;}public OrderStatus getOrderStatus() {return orderStatus;}public void setOrderStatus(OrderStatus orderStatus) {this.orderStatus = orderStatus;}}

 

编写取消订单接口与实现类

/*** 处理过期订单* @param orderId*/
void execExpiredOrder(Long orderId);
/*** 根据订单Id 修改订单的状态* @param orderId* @param processStatus*/
void updateOrderStatus(Long orderId, ProcessStatus processStatus);

 

@Override
public void execExpiredOrder(Long orderId) {// orderInfoupdateOrderStatus(orderId, ProcessStatus.CLOSED);
}
@Override
public void updateOrderStatus(Long orderId, ProcessStatus processStatus) {OrderInfo orderInfo = new OrderInfo();orderInfo.setId(orderId);orderInfo.setProcessStatus(processStatus.name());orderInfo.setOrderStatus(processStatus.getOrderStatus().name());orderInfoMapper.updateById(orderInfo);
}

 

 

最终会变成Close 

相关文章:

尚品汇-延迟插件实现订单超时取消(四十五)

目录&#xff1a; &#xff08;1&#xff09;延迟插件封装 &#xff08;2&#xff09;基于延迟插件测试 如何保证消息幂等性&#xff1f; &#xff08;3&#xff09;改造订单service-order模块-实现订单超时取消 &#xff08;1&#xff09;延迟插件封装 把消息带过去&#…...

欺诈文本分类检测(十一):LLamaFactory多卡微调

1. 引言 前文训练时都做了一定的编码工作&#xff0c;其实有一些框架可以支持我们零代码微调&#xff0c;LLama-Factory就是其中一个。这是一个专门针对大语言模型的微调和训练平台&#xff0c;有如下特性&#xff1a; 支持常见的模型种类&#xff1a;LLaMA、Mixtral-MoE、Qw…...

SprinBoot+Vue健康管管理微信小程序的设计与实现

目录 1 项目介绍2 项目截图3 核心代码3.1 Controller3.2 Service3.3 Dao3.4 application.yml3.5 SpringbootApplication3.5 Vue3.6 uniapp代码 4 数据库表设计5 文档参考6 计算机毕设选题推荐7 源码获取 1 项目介绍 博主个人介绍&#xff1a;CSDN认证博客专家&#xff0c;CSDN平…...

C++基础类容详解

目录 知识点1 C的概述 1 C的特征 2 C程序的编辑、编译和执行 3 第一个C源程序 4 面向对象程序设计思想 4.1 面向对象程序设计思想初始 4.2 面向对象程序设计思想的核心 知识点2 C对C的扩展 1 作用域访问运算符(::) 2 名称空间域 2.1 创建名称空间域 2.2 已有名称空间…...

python基础(16面试题附答案一)

python系列文章目录 python基础&#xff08;01变量&数据类型&运算符&#xff09; python基础&#xff08;02序列共性&#xff09; python基础(03列表和元组) python基础&#xff08;04字符串&字典&#xff09; python基础&#xff08;05集合set&#xff09; pytho…...

Leetcode3256. 放三个车的价值之和最大 I

Every day a Leetcode 题目来源&#xff1a;3256. 放三个车的价值之和最大 I 解法1&#xff1a;贪心 从大到下排序矩阵所有值, 记为数组v。 转化此题&#xff1a;从r*c个数中选取3个数分别给到车1&#xff0c;车2&#xff0c;和车3&#xff0c;使得符合条件的三数之和最大。…...

Redis中String类型的基本命令

文章目录 一、String字符串简介二、常见命令setgetmgetmsetsetnxincrincrbydecrdecrbyincrbyfloatappendgetrangesetrangestrlen 三、命令小结四、字符串内部编码五、String典型使用场景1. 缓存(Cache)功能2. 计数功能3. 共享会话&#xff08;Session&#xff09;4. 手机验证码…...

2024 年高教社杯全国大学生数学建模竞赛题目【A/B/C/D/E题】完整思路

↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑ A题是数模类赛事很常见的物理类赛题&#xff0c;需要学习不少相关知识。此题涉及对一个动态系统的建模&#xff0c;模拟…...

HR招聘新员工,如何考察企业文化适配度

要解决文化适配性问题&#xff0c;那在招聘过程中一定要明确企业核心价值观。比如通过制定明确文化价值观手册的方式&#xff0c;向求职者展示企业的使命愿景和价值观。 目前最为理想的考察方式就是线上的人才测评&#xff0c;比如&#xff1a;采用职业价值观测评法&#xff0…...

AI算力「搅局」座舱SoC

对于芯片巨头来说&#xff0c;汽车是难以割舍的赛道。 这不仅仅是因为「车规级」向来是准入门槛最高的细分市场之一&#xff0c;更重要的原因来自于从PC、智能手机到智能汽车时代&#xff0c;芯片公司都在寻求成为新周期的标杆。 比如&#xff0c;从PC时代的「英特尔」到智能手…...

lvs DR模式调试

DS配置&#xff1a; # cat /etc/keepalived_docker/keepalived.conf ! Configuration File for keepalived global_defs {router_id LVS_70 # 设置lvs的id&#xff0c;在一个网络内应该是唯一的 }vrrp_instance VI_70 {state MASTER # 两个 DS&#xff0…...

Java线程池的优化策略与最佳实践

哈喽&#xff0c;各位小伙伴们&#xff0c;你们好呀&#xff0c;我是喵手。运营社区&#xff1a;C站/掘金/腾讯云/阿里云/华为云/51CTO&#xff1b;欢迎大家常来逛逛 今天我要给大家分享一些自己日常学习到的一些知识点&#xff0c;并以文字的形式跟大家一起交流&#xff0c;互…...

android面试:解释一下 AsyncTask是什么?

AsyncTask 是 Android 中用于处理异步操作的一个类&#xff0c;它允许在后台线程中执行任务&#xff0c;并在完成后将结果传递回主线程。AsyncTask 主要用于执行短时间的后台操作&#xff0c;例如网络请求、文件读写等&#xff0c;而不阻塞用户界面。 AsyncTask 的主要特点&am…...

Django+Vue3前后端分离学习(四)(登录功能实现)

1、序列化数据&#xff1a; 创建serializers.py的python文件 从rest_framework里导入serializers类&#xff1a; from rest_framework import serializers class LoginSerializer(serializers.Serializer):email serializers.EmailField(requiredTrue, error_messages{&qu…...

机器学习面试:SVM为什么使用对偶函数求解?

支持向量机&#xff08;SVM&#xff09;在求解过程中使用对偶函数的原因主要与优化问题的性质、计算效率以及模型的泛化能力有关。以下是对偶函数在 SVM 中使用的详细解释&#xff1a; 1. 原始问题与对偶问题 在 SVM 中&#xff0c;我们的目标是找到一个超平面来最大化分类间…...

RabbitMQ 入门教程

介绍 RabbitMQ 是一个消息中间件&#xff0c;它实现了 AMQP (Advanced Message Queuing Protocol) 协议。本教程将引导你通过几个简单的步骤来学习如何使用 RabbitMQ 发送和接收消息。 环境准备 1. 安装 RabbitMQ - 在你的系统上安装 RabbitMQ: https://www.rabbitmq.com/d…...

docker进阶 compose等

Docker Compose 简介&#xff1a; 比如有100个微服务&#xff0c;不需要手动启动每一个&#xff0c;可以使用docker compose定义运行多个容器&#xff0c;高效管理化。 定义、运行多个容器 YAML file配置文件 single command 命令 写docker-compose.yaml docker-compose …...

[详细建模已更新]2024数学建模国赛高教社杯A题:“板凳龙” 闹元宵 思路代码文章助攻手把手保姆级

A 题 “板凳龙” 闹元宵 “板凳龙”&#xff0c;又称“盘龙”&#xff0c;是浙闽地区的传统地方民俗文化活动。人们将少则几十条&#xff0c;多则上百条的板凳首尾相连&#xff0c;形成蜿蜒曲折的板凳龙。盘龙时&#xff0c;龙头在前领头&#xff0c;龙身和龙尾相随盘旋&#x…...

网络编程(TCP+网络模型)

【1】TCP 初版服务器 #include <stdio.h> #include <sys/types.h> /* See NOTES */ #include <sys/socket.h> #include <netinet/in.h> #include <netinet/ip.h> #include <unistd.h> #include <arpa/inet.h> #include <string.h…...

Docker Image 命令

文章目录 目录 文章目录 1 . Docker镜像是什么? 2 . 镜像命令详解 docker images docker tag docker pull docker rmi docker save 总结 1 . Docker镜像是什么? Docker image 本质上是一个 read-only 只读文件&#xff0c; 这个文件包含了文件系统、 源码、库文件…...

AB测试中的因果推断陷阱:为什么你的随机化试验可能不靠谱?

AB测试中的因果推断陷阱&#xff1a;为什么你的随机化试验可能不靠谱&#xff1f; 在电商大促期间&#xff0c;某平台将"满200减30"的优惠券随机发放给50%用户&#xff0c;一周后发现实验组GMV提升12%&#xff0c;看似效果显著。但进一步分析发现&#xff0c;实验组中…...

GCC优化禁用指南:精准控制编译行为的5种方法

1. 为什么需要禁用GCC优化&#xff1f; 在嵌入式开发或者调试过程中&#xff0c;我们经常会遇到一些奇怪的bug&#xff1a;明明代码逻辑没有问题&#xff0c;但程序运行时却出现异常。这时候很可能就是编译器优化在"捣鬼"。GCC作为最常用的开源编译器&#xff0c;它的…...

AI FUTURE北京亦庄AI未来大会在京启幕

4月8日&#xff0c;AI FUTURE北京亦庄AI未来大会在北京经济技术开发区&#xff08;简称“北京经开区”&#xff0c;又称“北京亦庄”&#xff09;北京智慧电竞赛事中心正式启幕。这场为期两天的人工智能盛会&#xff0c;以“让每个人看AI的另一面”为主题&#xff0c;集高端论坛…...

53、竞态条件和同步---------多线程、竟态条件和同步

竞态条件和同步线程是程序执行的最小单位&#xff0c;一个进程可以包含多个线程&#xff0c;多个线程共享进程的资源&#xff08;如内存空间&#xff09;。在多线程环境中&#xff0c;线程之间的并发执行可能导致对共享资源的竞争。 竞态条件&#xff08;Race Condition&#x…...

Ecqlipse32:车规级嵌入式LCD显示驱动框架

1. 项目概述Ecqlipse32 是一款专为大众汽车集团 CARIAD 车载信息娱乐系统&#xff08;IVI&#xff09;平台定制开发的嵌入式 TFT-LCD 显示驱动框架&#xff0c;面向基于 ARM Cortex-M 系列微控制器&#xff08;特别是 STM32H7 和 NXP i.MX RT117x 等高性能 MCU&#xff09;的车…...

从《节奏医生》到你的游戏:拆解Koreographer Pro版如何实现高级音频集成(Wwise/FMOD)

从《节奏医生》到你的游戏&#xff1a;Koreographer Pro版如何实现高级音频集成&#xff08;Wwise/FMOD&#xff09; 在《节奏医生》这类音游中&#xff0c;玩家按键与音乐节拍的完美同步是游戏体验的核心。这种精准的音频同步背后&#xff0c;往往需要复杂的音频中间件集成。对…...

化工巡检机器人

山东奇妙智能科技有限公司专注于化工行业智能巡检机器人的研发与应用&#xff0c;其产品旨在通过自动化、智能化技术替代传统人工巡检&#xff0c;提升化工生产环境的安全性、效率和精准度。该类机器人通常具备防爆设计、多传感器融合、自主导航等功能&#xff0c;适用于易燃易…...

5年数据开发转AI Agent|30天学习路线

✅给想转的数开兄弟姐妹的真心话&#xff1a; 别裸辞&#xff01;先试水再转&#xff1a;利用业余时间学基础、做1-2个小Demo&#xff0c;验证自己是否真的感兴趣&#xff0c;再决定要不要all in&#xff1b; 把数开经验变成你的护城河&#xff1a;面试别只说你学了什么Agent框…...

Shell应用手册(一) 3.Linux环境搭建全攻略:虚拟机/云服务器/本地容器三种方式全覆盖

对于程序员、运维工程师或Linux学习者而言&#xff0c;搭建一个稳定、高效的Linux环境是开展工作和学习的基础。目前主流的搭建方式主要有三种&#xff1a;虚拟机&#xff08;适合本地学习练手&#xff09;、云服务器&#xff08;适合线上部署、远程访问&#xff09;、本地容器…...

YOLO+SAM工业缺陷检测:从理论到落地的完整方案

YOLOSAM工业缺陷检测&#xff1a;从理论到落地的完整方案一、痛点 PCB质检中&#xff0c;人工标注缺陷边界8分钟/张。YOLO框不准&#xff0c;SAM对工业缺陷水土不服。 二、解决方案 LoRA微调SAM&#xff1a;只改2%参数&#xff0c;速度3倍提升&#xff0c;显存24GB→8GB。 Dice…...