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

Redis第18讲——Redis和Redission实现延迟消息

即使不是做电商业务的同学,也一定知道订单超时关闭这种业务场景,这个场景大致就是用户下单后,如果在一定时间内未支付(比如15分钟、半小时),那么系统就会把这笔订单给关闭掉。这个功能实现的方式有很多种,比如JDK中自带的DelayQueue延迟队列、Timer、ScheduledThreadPoolExecutor,强烈推荐的RocketMQ、RabblitMQ及Kafka等消息队列,还有就是Hutool的SystemTimer、Netty的HashedWheelTimer等等,感兴趣的可以去了解一下今天我们就先看看Redis和Redisson是如何实现延迟消息的。

一、Redis如何实现延迟消息

1.1 过期key监听

很多人都知道Redis有一个过期监听的功能,在redis.conf中加一条notify-keyspace-events开启过期监听,然后在代码中实现KeyExpirationMessageListener就可以监听key的过期消息了。

这样就可以在接收到过期消息的时候进行关单操作了,但是这个方案并不推荐,Redis官方明确说过Redis并不保证key在过期的时候就能被立即删除,更不保证这个消息能被立即发出,所以,消息延迟是必然的,数据量越大延迟的时间越长。

而且,在Redis5.0之前,这个消息是通过PUB/SUB模式发出的,不会做持久化,至于你有没有接收到,有没有消费成功,它不管,所以,如果发消息的时候客户端挂了,之后再恢复的话,这个消息也就彻底丢了。

1.2 Zset

我们可以借助Redis中的有序集合——zset来实现这个功能,zset是一个有序集合,每一个元素(member)都关联了一个score,可以通过score排序来取集合中的值。

我们可以将订单超时时间的时间戳(下单时间+超时时间)作为score订单号orderId作为成员(member),这样redis会对zset按照score进行排序,再通过定时任务获取“当前时间>=score"的延迟任务,获取到之后就可以根据订单号(member)进行关单操作。

这么实现的优点就是可以借助redis持久化和高可用机制,避免数据丢失。

1.3 zset实现超时关单Demo

步骤:

  • 用户下单后,将信息写入redis zset缓存中,score为当前时间+延迟时间的时间戳,member为订单号。

  • 使用ZRangeByScore和WithScores命令,获取当前时间戳之前的所有任务,并通过score判断哪些任务已到期,进行关单处理。

  • 启动一个额外的定时任务周期性检查并处理已到期的订单。

导入依赖:

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

Demo:

@Service
public class OrderDelayService {@Autowiredprivate RedisTemplate<String,String> redisTemplate;//延迟15分钟private final long delayTime = 15*60*1000;//订单号key前缀private static final String ORDER_KEY="order:";//延迟关单keyprivate static final String DELAY_KEY="close_orders";/*** 创建订单* @param orderId*/public void createOrder(String orderId){//...//创建订单成功//1.获取当前时间戳long currentTime = System.currentTimeMillis();//2.score:当前时间+延迟时间long score=currentTime+delayTime;//3.加入redis zset集合redisTemplate.opsForZSet().add(DELAY_KEY,  //redis keyORDER_KEY+orderId, //memberscore //score);}
​/*** 任务间隔一秒执行一次*/@Scheduled(fixedDelay = 1000)public void closeExpiredOrders(){//当前时间戳long currentTime = System.currentTimeMillis();ZSetOperations<String, String> zSetOps = redisTemplate.opsForZSet();//取出所有数据(已排好序)Set<String> orderKeys = zSetOps.range(DELAY_KEY, 0, -1);for (String orderKey : orderKeys) {//获取score,double score = zSetOps.score(DELAY_KEY, orderKey);if(currentTime>=score){String orderId = orderKey.substring(ORDER_KEY.length());//进行关单操作...closeOrder(orderId)//从zset里移除该订单zSetOps.remove(DELAY_KEY,orderKey);}}}
}

二、Redission如何实现延迟消息

2.1 实现原理

Redission中定义了分布式延迟队列DelayedQueue,其实就是在zset的基础上增加了一个基于内存的延迟队列。

大致的流程如下:

当我们要添加一个数据到延迟队列的时候,redission会把数据+超时时间放到zset中,并且起一个延时任务,当任务到期的时候,再去zset中把数据取出来,返回给客户端使用。

2.2 案例

导入依赖:

 <dependency><groupId>org.redisson</groupId><artifactId>redisson-spring-boot-starter</artifactId><version>3.13.1</version>
</dependency>

定义一个客户端:

@Component
public class RedissionConfig {@Bean(destroyMethod = "shutdown")public RedissonClient redssion(){Config config=new Config();config.useSingleServer().setAddress("redis://127.0.0.1");return Redisson.create(config);}
}

实现:

@Component
public class RedissionDelay {
​@Autowiredprivate RedissonClient client;/*** 创建订单,并设置过期时间* @param orderId*/public void createOrder(String orderId){//...//创建订单成功RBlockingDeque<Object> blockingDeque = client.getBlockingDeque("orderQueue");RDelayedQueue<Object> delayedQueue = client.getDelayedQueue(blockingDeque);//将订单加入延迟队列,延迟时间为15分钟delayedQueue.offer(orderId,15, TimeUnit.MINUTES);}
​/*** 关单操作*/public void closeOrder(){RBlockingDeque<Object> orderQueue = client.getBlockingDeque("orderQueue");new Thread(() -> {while (true){try {String orderId = (String) orderQueue.take();//进行关单操作,closeOrder(orderId)} catch (InterruptedException e) {e.printStackTrace();}}});}
}

上述例子,我们用RDelayedQueue的offer方法将订单添加到了延迟队列,并指定了延迟时间,当元素的延迟时间到达时,Redission会将元素从RDelayedQueue转移到与之关联的RBlockingDeque。

然后在检查是否要关单的时候,另起了一个线程,不断循环读取到期的订单。值得注意的是 take方法从RBlockingDeque中获取元素,这是一个阻塞操作,如果没有元素,它会一直等到,直到有元素。

End:希望对大家有所帮助,如果有纰漏或者更好的想法,请您一定不要吝啬你的赐教🙋。

相关文章:

Redis第18讲——Redis和Redission实现延迟消息

即使不是做电商业务的同学&#xff0c;也一定知道订单超时关闭这种业务场景&#xff0c;这个场景大致就是用户下单后&#xff0c;如果在一定时间内未支付&#xff08;比如15分钟、半小时&#xff09;&#xff0c;那么系统就会把这笔订单给关闭掉。这个功能实现的方式有很多种&a…...

返回枚举类给前端

1. 前言 在实际开发过程中&#xff0c;前端的下拉框或者单选按钮的内容通常的需要和后端匹配的&#xff0c;故一般会由后端将下拉框的内容或单选框的内容传给前端&#xff0c;而这些内容在后端一般是由枚举类存储的&#xff0c;如果后端直接返回枚举类&#xff0c;返回结果将会…...

A. Maximize?

time limit per test 2 seconds memory limit per test 256 megabytes input standard input output standard output You are given an integer x&#x1d465;. Your task is to find any integer y&#x1d466; (1≤y<x)(1≤&#x1d466;<&#x1d465;) su…...

RBAC 动态权限

文章目录 前言一、RBAC&#xff08;Role-Based Access Control&#xff0c;基于角色的访问控制&#xff09;二、Java实现RBAC 权限的大概思路1. 添加依赖2. 配置MyBatis-Plus和数据源1. 添加依赖2. 实体类与Mapper接口UserMapper.java 3. 配置MyBatis-Plus4. 自定义UserDetails…...

c语言:模拟strlen(三种方法)最全版本

1.计数的方法 #include <stdio.h> #include <assert.h> int my_strlen(const char * str)//const的使用优化 {int count0;assert(str)while(*str){count;str;}return count; } 2.用指针的方法&#xff08;指针-指针&#xff09; #include <stdio.h> #incl…...

线性模型--普通最小二乘法

线性模型 一、模型介绍二、用于回归的线性模型2.1 线性回归&#xff08;普通最小二乘法&#xff09; 一、模型介绍 线性模型是在实践中广泛使用的一类模型&#xff0c;该模型利用输入特征的线性函数进行预测。 二、用于回归的线性模型 以下代码可以在一维wave数据集上学习参…...

移动云以深度融合之服务,令“大”智慧贯穿云端

移动云助力大模型&#xff0c;开拓创新领未来。 云计算——AI模型的推动器。 当前人工智能技术发展的现状和趋势&#xff0c;以及中国在人工智能领域的发展策略和成就。确实&#xff0c;以 ChatGPT 为代表的大型语言模型在自然语言处理、文本生成、对话系统等领域取得了显著的…...

簡述vue常用指令

Vue.js 提供了许多内置指令&#xff0c;这些指令用于在模板中添加特殊功能。以下是一些 Vue 的常用内置指令的简要说明&#xff1a; v-text&#xff1a; 更新元素的 textContent。示例&#xff1a;<span v-text"message"></span> v-html&#xff1a; 更…...

【建议收藏】用AI快速生成一个网页(名侦探柯南~灰原哀主题网页),适合大学生web期末大作业

下面是提供给AI的提示词和AI给出的代码以及成果展示 1、生成一个网页导航栏&#xff0c;宽度为1300px&#xff0c;高度为60px。导航区域在导航栏最右侧不超出导航栏&#xff0c;高60px&#xff0c;宽度500px&#xff0c;里面是5个导航菜单项横向排列&#xff0c;每个宽度100px&…...

用c++用4个凸函数(觉得啥好用用啥)去测试adam,rmsprop,adagrad算法的性能(谁先找到最优点)

为了测试 Adam、RMSProp 和 Adagrad 算法的性能&#xff0c;你可以使用四个凸函数进行实验。以下是一些常用的凸函数示例&#xff1a; Rosenbrock 函数&#xff1a; Booth 函数&#xff1a; Himmelblau 函数&#xff1a; Beale 函数&#xff1a; 你可以选择其中一个或多…...

AJAX初级

AJAX的概念&#xff1a; 使用浏览器的 XMLHttpRequest 对象 与服务器通信 浏览器网页中&#xff0c;使用 AJAX技术&#xff08;XHR对象&#xff09;发起获取省份列表数据的请求&#xff0c;服务器代码响应准备好的省份列表数据给前端&#xff0c;前端拿到数据数组以后&#xf…...

重载大于号运算符,比较复数大小

本题目要求编写代码的功能为&#xff1a; 输入两个复数&#xff08;变量名自拟&#xff09;&#xff0c;比较复数模的大小&#xff0c;复数实部与虚部都是整数 要求输入时输入4个整数&#xff0c;分别代表复数1的实部、虚部&#xff0c;复数2的实部虚部 输入格式: 在同一行中输…...

go ast语义分析实现指标计算器

什么是AST 首先我们要知道AST是什么&#xff08;Abstract Syntax Tree&#xff0c;AST&#xff09;&#xff0c;简称为语法树&#xff0c;是go语言源代码语法结构的一种抽象表示。它以树状的形式表现编程语言的语法结构&#xff0c;树上的每个节点都表示源代码中的一种结构。 …...

【Vue】组件间传参与方法调用

【前言】 … 【目标】 1 了解组件间传参 2 组件间自定义事件绑定与解绑 3 组件的事件总线 4 消息订阅与发布 一 组件间传参 1 props 引入并使用组件:传递参数 <template><div id="app"><HelloWorld :msg="msg" :name="name" …...

类和对象2

三、C对象模型和this指针 3.1 成员变量和成员函数分开存储 在C中&#xff0c;类内的成员变量和成员函数分开存储&#xff0c;只有非静态成员变量才属于类的对象上 #define _CRT_SECURE_NO_WARNINGS 1 #include <iostream> #include <string.h> using namespace …...

Linux系统命令traceroute详解(语法、选项、原理和实例)

目录 一、traceroute概述 二、语法 1、基本语法 2、命令选项 三、帮助信息 四、示例 1. 使用默认模式&#xff08;ICMP Echo&#xff09;追踪到目标主机 2. 使用UDP模式&#xff08;需要root权限&#xff09;追踪到目标主机 3. 不解析IP地址为主机名&#xff0c;直接显…...

中兴通讯助力中国移动,推动SPN AI节能技术于23省规模部署

SPN作为中国移动自主创新的新一代综合承载网络&#xff0c;相比PTN设备&#xff0c;SPN的单机容量及性能有大幅提升&#xff0c;整机功耗也相应变大。在当前国家双碳政策的目标下&#xff0c;SPN设备的节能降耗也日益成为中国移动关注的焦点。因此&#xff0c;中国移动选择与中…...

SQL Server--死锁

今天&#xff0c;客户反应打不开xxx页面了。好家伙肯定锁表了。。。。。 只能先吧死锁进程先kill掉&#xff0c;不能耽误客户生产环境运行。。。。。 一定要看看是那张表发生了死锁 1、查询死锁语句 select dbid,* from sys.sysprocesses where 11 and spid >50 and blo…...

中科蓝讯AB32VG1中文寄存器说明GPIO端口操作

1 GPIO管理 1.1 GPIO通用控制寄存器 寄存器 1- 1 GPIOA&#xff1a;端口 A 数据寄存器 位寄存器名模式缺省描述31:8---未使用7:0GPIOA写0x00PAx 数据。当 PAx 用作 GPIO 时有效 0&#xff1a;读取时PAx为输入低电平状态&#xff0c;写入时PAx为输出低电平; 1&#xff1a;PAx…...

如何查看热门GPT应用?

1、登陆chatgpt 2、访问 https://chatgpt.com/gpts 3、在该界面&#xff0c;可以搜索并使用image generator, Write For Me&#xff0c;Language Teature等热门应用。...

QQ群数据采集终极教程:5分钟掌握批量抓取技巧

QQ群数据采集终极教程&#xff1a;5分钟掌握批量抓取技巧 【免费下载链接】QQ-Groups-Spider QQ Groups Spider&#xff08;QQ 群爬虫&#xff09; 项目地址: https://gitcode.com/gh_mirrors/qq/QQ-Groups-Spider 还在为手动收集QQ群信息而烦恼吗&#xff1f;QQ-Groups…...

揭秘CuCl超低热导率:四声子散射与温度重正化的关键作用

1. 项目概述&#xff1a;为何要深挖CuCl的热导率&#xff1f;在材料科学和凝聚态物理的交叉领域&#xff0c;热输运性质的研究从来都不是一个孤立的课题。它直接关系到热电材料的转换效率、电子器件的散热能力&#xff0c;以及热障涂层的服役寿命。传统上&#xff0c;我们倾向于…...

机器学习预测细菌耐药性:从全基因组数据到公共卫生预警

1. 项目概述与核心价值抗菌药物耐药性&#xff08;AMR&#xff09;这事儿&#xff0c;现在谁提起来都头疼。它不再是实验室报告上的一个数字&#xff0c;而是直接关系到我们每个人生病了还有没有药可用的现实问题。弯曲杆菌&#xff0c;这个听起来有点拗口的名字&#xff0c;其…...

智慧树自动刷课插件终极指南:3步安装教程,彻底告别手动刷课烦恼!

智慧树自动刷课插件终极指南&#xff1a;3步安装教程&#xff0c;彻底告别手动刷课烦恼&#xff01; 【免费下载链接】zhihuishu 智慧树刷课插件&#xff0c;自动播放下一集、1.5倍速度、无声 项目地址: https://gitcode.com/gh_mirrors/zh/zhihuishu 还在为智慧树平台的…...

高光谱成像与机器学习:LDA+SVM/KNN实现蜂蜜植物源精准鉴别

1. 项目概述&#xff1a;当高光谱成像遇上机器学习&#xff0c;如何为蜂蜜“验明正身”&#xff1f;在食品行业&#xff0c;尤其是像蜂蜜这样的高价值农产品领域&#xff0c;“真实性”一直是消费者和生产者共同关注的焦点。一瓶标着“新西兰麦卢卡”或“东北椴树蜜”的蜂蜜&am…...

《AI推理优化实战:从高延迟高成本到高效低耗,企业级AI落地必备技术》

随着大模型、AI应用规模化落地&#xff0c;行业发展重心已经从“模型训练”全面转向“模型推理”。2026年AI产业的核心痛点不再是模型训练精度不足&#xff0c;而是推理成本过高、响应延迟过长、算力资源浪费。很多企业落地AI应用时&#xff0c;面临大模型推理速度慢、并发量低…...

贝叶斯优化驱动量子噪声建模:数据高效提升NISQ仿真精度

1. 项目概述与核心价值在量子计算领域&#xff0c;尤其是在当前嘈杂中型量子&#xff08;NISQ&#xff09;时代&#xff0c;一个核心的挑战在于如何让仿真结果与真实硬件上的运行结果保持一致。我们设计量子算法、优化电路布局、评估性能&#xff0c;很大程度上依赖于对量子处理…...

2026财务分析师新人如何快速提升能力:从“账房先生”到“战略参谋”的跃迁之路

2026年的财务领域&#xff0c;规则已经彻底改写。传统的“记账、算账、报账”正在被RPA和AI快速替代&#xff0c;企业真正渴求的&#xff0c;不再是只会做账的“会计”&#xff0c;而是能够深入业务、洞察数据、驱动决策的财务分析师。这不仅是职业进阶的必然要求&#xff0c;更…...

2026年照片去水印免费软件保姆级教程!学会这几招,告别水印烦恼

你是不是也遇到过这样的抓狂时刻&#xff1f;在平台上刷到一张特别适合做壁纸或配图的高清照片&#xff0c;兴冲冲地保存下来&#xff0c;结果角落里的水印瞬间让整张图的格调打了对折&#xff1b;又或者&#xff0c;自己辛辛苦苦做好的图片&#xff0c;在分享转发几道后&#…...

XC161芯片ULINK调试连接问题解决方案

1. ULINK与XC161 AC Step连接问题解析最近在调试XC161&#xff08;AC Step&#xff09;芯片时&#xff0c;遇到了一个典型问题&#xff1a;使用Keil ULINK USB-JTAG适配器无法建立连接&#xff0c;但同样的设备在Infineon XC161 Starter Kit&#xff08;AB Step&#xff09;上却…...