Spring Boot中使用Redis和Lua脚本实现延时队列

码到三十五 : 个人主页

延时队列是一种常见的需求。延时队列允许我们延迟处理某些任务,这在处理需要等待一段时间后才能执行的操作时特别有用,如发送提醒、定时任务等。文中,将介绍如何在Spring Boot环境下使用Redis和Lua脚本来实现一个延时队列。

目录

    • 一、延迟队列的四大使用场景
    • 二、如何利用ZSet实现延迟队列
    • 三、实现步骤
    • 四、实现代码
    • 五、使用ZSet实现延迟队列的缺陷
    • 六、替代实现方案
      • 结语

一、延迟队列的四大使用场景

  1. 订单超时自动处理
    在电商领域,延迟队列对于处理订单超时问题至关重要。一旦用户下单,订单信息便进入延迟队列,并预设超时时长。若用户在此时间内未完成支付,订单信息将由消费者从队列中提取,并执行如取消订单、库存释放等后续操作,高效且自动化。

  2. 优惠券到期温馨提醒
    借助延迟队列,我们可以实现优惠券到期前的温馨提醒服务。将临近过期的优惠券信息入队,并设定精确延迟时间。时间一到,系统自动提醒用户优惠券的到期日,引导他们及时享用优惠,提升用户体验。

  3. 智能消息重试策略
    在处理网络请求失败、数据库异常等情况时,延迟队列提供了智能的消息重试机制。当消息初次处理失败,它会被置入队列并设定重试延时。延时结束后,系统会再次尝试处理,确保消息的可靠传递与处理。

  4. 异步通知与定时提醒
    延迟队列还能用于实现异步通知和定时提醒功能。用户完成操作后,系统将相关通知信息加入队列,并设定发送延时,确保在最佳时机向用户推送通知,既不打扰用户,又能保持信息的时效性。

二、如何利用ZSet实现延迟队列

Redis的ZSet(有序集合)是一个根据分数对唯一字符串成员进行排序的数据结构。在多个成员分数相同时,它们会按照字典顺序进行排列。ZSet不仅常用于排行榜和限速器等场景,还可巧妙用于实现延迟队列。

在这里插入图片描述

基于ZSet的延迟队列实现原理,主要利用了其有序性和按分数排序的特点。以下是具体实现步骤的简要介绍:

  1. 定义延迟消息:在ZSet中,我们将延迟消息作为成员,而其对应的延迟时间则作为该成员的分数。这里的延迟时间通常是一个未来的时间戳,它指明了消息应当被处理的确切时刻。

  2. 消息入队:使用ZADD命令,我们可以轻松地将消息添加到ZSet中,并为其指定相应的延迟时间作为分数。

  3. 定期检查:通过定期轮询ZSet,我们可以利用ZRANGEBYSCORE命令来检索那些分数(即延迟时间)小于或等于当前时间戳的消息,这些消息即为到期的、需要被处理的消息。

  4. 消息处理与出队:一旦找到到期的消息,我们可以使用ZPOPMIN命令将它们从ZSet中移除,并进行相应的处理。在处理过程中,需要考虑并发性和数据一致性问题,确保每条消息都能被正确处理且不会被重复处理。

  5. 后续操作与通知:为了提高系统的性能和可靠性,我们可以结合Redis的Pub/Sub机制。在处理完消息后,发布一个事件来通知其他服务或订阅者进行后续的操作或处理。

通过这种方式,ZSet能够有效地按照消息的延迟时间顺序,逐个取出并处理到期的消息,从而实现了一个高效且可靠的延迟队列系统。

三、实现步骤

在这里插入图片描述

在Spring Boot环境下,实现一个基于Redis和Lua脚本的延时队列,需要以下几个步骤:

  1. 环境准备

    • 安装并启动Redis服务器。
    • 在Spring Boot项目中添加spring-boot-starter-data-redis依赖。
  2. Redis数据结构选择

    • 使用Redis的zset(有序集合)数据结构来存储延时任务。zset中的元素是唯一的,但分数(score)可以相同,可以用作任务的延迟时间戳。
  3. Lua脚本编写

    • 编写一个Lua脚本来处理队列的出队和入队操作,以确保操作的原子性。
  4. Spring Boot应用配置

    • 配置Redis连接工厂和Redis模板。
  5. 实现延时队列服务

    • 提供一个服务来管理延时队列,包括入队、出队、检查并处理到期的任务等。
  6. 定时任务调度

    • 使用Spring的@Scheduled注解或者Redis的键空间通知来定期检查并处理到期的任务。

四、实现代码

下面是一个简化版本的实现:

1. 添加Maven依赖

pom.xml中添加spring-boot-starter-data-redis依赖:

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

2. 配置Redis

application.ymlapplication.properties中配置Redis连接信息:

spring:redis:host: localhostport: 6379

3. Lua脚本
定义一个Lua脚本原子性地执行出队操作。脚本使用Redis的有序集合命令来查找并移除到期的任务:

-- KEYS[1] 延时队列的key
-- ARGV[1] 当前时间戳
-- 返回值:任务ID(如果存在)或nil
local key = KEYS[1]
local currentTime = tonumber(ARGV[1])
local task = redis.call('zrangebyscore', key, 0, currentTime, 'LIMIT', 0, 1)
if #task > 0 thenredis.call('zremrangebyscore', key, 0, currentTime)return task[1]
elsereturn nil
end

可以稍微优化一下上面的Lua脚本,以减少不必要的操作和提高效率:

-- KEYS[1] 延时队列的key
-- ARGV[1] 当前时间戳
-- 返回值:任务ID(如果存在)或nillocal key = KEYS[1]
local currentTime = tonumber(ARGV[1])-- 使用zrangebyscore和zrem的组合命令zpopmin,它原子性地返回并移除分数最低的元素
-- zpopmin命令(5.0及以上版本)
local task = redis.call('zpopmin', key, 1, 'BLOCK', 0, 'SCORES')-- zpopmin返回的是一个包含两个元素的数组,第一个元素是分数,第二个是成员
if task and #task > 0 and task[2] and tonumber(task[1]) <= currentTime thenreturn task[2] -- 返回任务ID
elsereturn nil
end

注意:

  1. zpopmin命令是一个原子性的操作,它返回并删除分数最低的元素。避免了先查询后删除可能带来的并发问题。zpopmin`命令在Redis 5.0及以上版本中可用。

  2. zpopmin命令可以设置阻塞时间,这里设置为0,表示不阻塞。如果希望在没有可用元素时阻塞等待一段时间,可以调整这个值。

  3. 脚本检查了返回的分数是否小于等于当前时间戳,以确保只处理到期的任务。

  4. 如果Redis版本低于5.0zpopmin将不可用,可以使用zrangebyscorezrem的组合,但需要注意并发问题。

4. 实现延时队列服务

@Service
public class DelayQueueService {@Autowiredprivate StringRedisTemplate stringRedisTemplate;private static final String DELAY_QUEUE_KEY = "delay_queue";// 入队操作public void enqueue(String taskId, long delayInSeconds) {long score = System.currentTimeMillis() / 1000 + delayInSeconds;stringRedisTemplate.opsForZSet().add(DELAY_QUEUE_KEY, taskId, score);}// 出队操作,使用Lua脚本确保原子性public String dequeue() {String luaScript = "..."; // 上面定义的Lua脚本内容RedisScript<String> script = RedisScript.of(luaScript, String.class);long currentTime = System.currentTimeMillis() / 1000;return stringRedisTemplate.execute(script, Collections.singletonList(DELAY_QUEUE_KEY), String.valueOf(currentTime));}
}

5. 定时任务调度

@Component
public class DelayQueueScheduler {@Autowiredprivate DelayQueueService delayQueueService;private static final long POLLING_INTERVAL = 1000; // 检查间隔1秒@Scheduled(fixedRate = POLLING_INTERVAL)public void pollAndProcess() {String taskId = delayQueueService.dequeue();if (taskId != null) {// 处理任务逻辑,例如调用某个服务或者方法等。System.out.println("Processing task: " + taskId);}}
}

五、使用ZSet实现延迟队列的缺陷

虽然Redis的ZSet能满足一些简单场景的延迟队列需求,但也存在一些明显的缺陷。

  1. 资源空转问题
    延迟任务的时间分布往往是不均匀的。在某些时段,可能会有大量的任务需要处理,而在其他时段则可能几乎没有任务。这种情况下,如果系统持续检查ZSet以寻找到期任务,那么在任务稀少或无任务的时段,系统会处于空转状态,这无疑是对计算资源的浪费。

  2. 性能瓶颈
    当延迟消息数量众多时,不断地轮询整个ZSet以查找到期消息会对性能产生显著影响。特别是当任务数量庞大且到期时间分散时,范围查询的开销会变得尤为突出。此外,如果多个任务同时到期且回调函数执行效率低下,还可能导致延迟处理中心的性能下降,进而引发连锁反应,影响到后续任务的及时处理。

  3. 时间精度问题
    ZSet使用浮点数作为分数来排序元素,这在某些需要高精度时间控制的场景中可能不够用。同时,Redis实例的故障、重启或时钟回拨等问题都可能影响到延迟事件处理的准确性。

六、替代实现方案

  1. 状态即时校验
    在某些业务流程中,可以通过即时校验当前状态与应有状态的方式来替代延迟队列。但这种方法更适用于工单等可以持续校验的业务场景,对于一次性的延迟通知任务则不太适用。

  2. 利用消息中间件的延迟消息功能
    像RocketMQ和RabbitMQ这样的消息中间件提供了延迟消息的功能。例如,RocketMQ在商业版本中支持自定义时长的延迟消息。

  3. 数据库轮询
    通过定期轮询数据库中的业务单据表或专门的延迟事件表来处理过期任务。但这种方法可能会对业务数据库和服务造成性能负担,且轮询的时间间隔难以精确把控。

  4. 时间轮算法
    时间轮算法是一种有效的处理定时任务的方法。但为了实现持久化和避免任务丢失,需要结合Redis或关系数据库来存储延迟任务。在服务启动时,需要将存储的延迟任务加载到时间轮中,并在任务过期后更新任务状态,以防止重复执行或加载。

结语

通过使用Redis和Lua脚本,可以在Spring Boot环境中实现一个高效且可靠的延时队列系统。这种方法利用了Redis的有序集合数据结构和Lua脚本的原子性操作来确保任务的正确性和一致性。通过定期调度任务来处理到期的任务,可以实现各种需要延迟执行的操作,如发送提醒、执行定时任务等。



听说...关注下面公众号的人都变牛了,纯技术,纯干货 !

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

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

相关文章

吴恩达机器学习笔记 三十七 电影推荐系统 使用特征 成本函数 协同过滤算法

以电影评分系统为例&#xff0c;令 r(i, j) 来表示用户 j 已经对电影 i 评分&#xff0c; y&#xff08;i, j&#xff09;表示评分具体是多少。 假如每部电影有自己的特征&#xff0c;那么用户 j 对电影 i 的评分预测为 w(j) * x(i) b(j) r(i, j) &#xff1a;一个用户 j 是否…

global IoT SIM解决方案

有任何关于GSMA\IOT\eSIM\RSP\业务应用场景相关的问题&#xff0c;欢迎W: xiangcunge59 一起讨论, 共同进步 (加的时候请注明: 来自CSDN-iot). Onomondo提供的全球IoT SIM卡解决方案具有以下特点和优势&#xff1a; 1. **单一全球配置文件**&#xff1a;Onomondo的SIM卡拥…

Flink checkpoint 源码分析- Checkpoint barrier 传递源码分析

背景 在上一篇的博客里&#xff0c;大致介绍了flink checkpoint中的触发的大体流程&#xff0c;现在介绍一下触发之后下游的算子是如何做snapshot。 上一篇的文章: Flink checkpoint 源码分析- Flink Checkpoint 触发流程分析-CSDN博客 代码分析 1. 在SubtaskCheckpointCoo…

Vue3+ts(day05:ref、props、生命周期、hook)

学习源码可以看我的个人前端学习笔记 (github.com):qdxzw/frontlearningNotes 觉得有帮助的同学&#xff0c;可以点心心支持一下哈&#xff08;笔记是根据b站上学习的尚硅谷的前端视频【张天禹老师】&#xff0c;记录一下学习笔记&#xff0c;用于自己复盘&#xff0c;有需要学…

开源AI智能名片S2B2C商城系统:移动技术的深度整合与应用

在数字化营销的新时代&#xff0c;开源AI智能名片S2B2C商城系统通过深度整合移动技术MAC ID、Beacon和DSP&#xff0c;为企业带来了前所未有的营销机遇。这一系统不仅提高了营销效率&#xff0c;还极大地提升了客户体验&#xff0c;并有效降低了营销成本。下面&#xff0c;我们…

【JAVA基础之反射】反射详解

&#x1f525;作者主页&#xff1a;小林同学的学习笔录 &#x1f525;mysql专栏&#xff1a;小林同学的专栏 1.反射 1.1 概述 是在运行状态中&#xff0c;对于任意一个类&#xff0c;都能够知道这个类的所有属性和方法&#xff1b; 对于任意一个对象&#xff0c;都能够调用它…

SQL注入漏洞扫描---sqlmap

what SQLMap是一款先进的自动执行SQL注入的审计工具。当给定一个URL时&#xff0c;SQLMap会执行以下操作&#xff1a; 判断可注入的参数。判断可以用哪种SQL注入技术来注入。识别出目标使用哪种数据库。根据用户的选择&#xff0c;读取哪些数据库中的数据。 更详细语法请参考…

BJFUOJ-C++程序设计-实验3-继承和虚函数

A TableTennisPlayer 答案&#xff1a; #include<iostream> #include<cstring> using namespace std;class TableTennisPlayer{ private:string firstname;string lastname;bool hasTable;public:TableTennisPlayer(const string &, const string &, bool…

leetCode65. 有效数字

leetCode65. 有效数字 题目思路 代码 class Solution { public:bool isNumber(string s) {int l 0, r s.size() - 1;// 1.忽略前后的空格while(l < r && s[l] ) l;while(l < r && s[r] ) r--;if(l > r) return false;s s.substr(l,r - l 1)…

Git的基本操作和使用

git分支指令 列出所有本地分支 git branchmaster是绿的 前面有个 表示当前分支是master* 列出所有远程分支 git branch -r列出所有本地分支和远程分支 git branch -a新建一个分支&#xff0c;但依然停留在当前分支 git branch [branch-name]新建一个分支&#xff0c;并切…

httpcanary抓包某游戏思路及教程[第1期]

游戏介绍&#xff1a; 这期在线读档0花购买教程&#xff0c;存档版教程。下一期在线购买无限鲜花累计充值安卓全系统适配修改教程。 小白勿入&#xff0c;技术流资料 网盘自动获取 链接&#xff1a;https://pan.baidu.com/s/1lpzKPim76qettahxvxtjaQ?pwd0b8x 提取码&#x…

设计模式动态代理

什么是设计模式? 一个问题通常有n种解法&#xff0c;其中肯定有一种解法是最优的&#xff0c;这个最优的解法被人总结出来了&#xff0c;称之为设计模式。 设计模式有20多种&#xff0c;对应20多种软件开发中会遇到的问题。 关于设计模式的学习&#xff0c;主要学什么&#…

onedrive下載zip檔案有20G限制,如何解決

一般來說&#xff0c;OneDrive網頁版對文件下載大小的限制如下圖所示&#xff0c;更多資訊&#xff0c;請您參考這篇文章&#xff1a;OneDrive 和 SharePoint 中的限制 - Microsoft Support 因此我們推薦您使用OneDrive同步用戶端來同步到本地電腦&#xff0c;您也可以選擇只同…

MFC 列表控件删除实例(源码下载)

1、本程序基于前期我的博客文章《MFC下拉菜单打钩图标存取实例&#xff08;源码下载) 》 2、程序功能选中列表控件某一项&#xff0c;删除按钮由禁止变为可用&#xff0c;点击删除按钮&#xff0c;选中的项将删除。 3、首先在主界面添加一个删除参数按钮。 4、在myDlg.cpp 文件…

巨人网络发布2023年年报:全力拥抱AI浪潮,开启游戏产业新篇章

易采游戏网5月3日消息&#xff0c;国内知名游戏公司巨人网络发布了其2023年度财务报告&#xff0c;报告显示&#xff0c;公司在过去一年中积极拥抱AI技术&#xff0c;实现了业绩的稳步增长&#xff0c;为游戏产业带来了新的活力与机遇。 在报告中&#xff0c;巨人网络详细阐述了…

【C++】STL简介

&#x1f525;个人主页&#xff1a; Forcible Bug Maker &#x1f525;专栏&#xff1a; C 目录 前言什么是STL&#xff1f;STL的历史STL的版本STL六大组件STL的优缺点STL的优点&#xff1a;STL的缺点&#xff1a; 如何学习STL结语 前言 本篇博客主要内容&#xff1a;STL简介。…

240503-关于Unity的二三事

240503-关于Unity的二三事 1 常用快捷键 快捷键描述CtrlP播放/停止Ctrl1打开Scene窗口Ctrl2打开Game窗口Ctrl3打开Inspect窗口Ctrl4打开Hierarchy窗口Ctrl5打开Project窗口Ctrl6打开Animation窗口 2 关联VisualStudio2022 3 节约时间&#xff1a;将最新声明的参数移动到最上…

解决wget报错:ERROR 403: Forbidden.

原因&#xff1a; 服务器正在检查引用者&#xff0c;部分 HTTP 请求会得到错误响应。不以 Mozilla 开头或不包含 Wget 的用户代理的请求会被拒绝。 解决方案&#xff1a; wget --user-agent“Mozilla” 要下载的链接 如&#xff1a; wget --user-agent"Mozilla" …

jvm垃圾回收机制介绍

JVM&#xff08;Java虚拟机&#xff09;是Java程序的运行环境&#xff0c;它负责执行字节码文件。JVM的工作原理主要包括以下几个部分&#xff1a;类加载器、执行引擎、垃圾收集器和内存管理。类加载器负责加载字节码文件并将其转换成Java平台上的机器码&#xff0c;执行引擎负…

java入门-日期类

日期类 Date类 Date类表示特定的时间&#xff0c;可以精确到毫秒。 获取Date对象 Date date new Date(); 构造方法 /*** Allocates a <code>Date</code> object and initializes it so that* it represents the time at which it was allocated, measured to…