0

0

Redis如何使用Lua脚本实现滑动窗口限流

P粉602998670

P粉602998670

发布时间:2026-03-18 10:21:34

|

516人浏览过

|

来源于php中文网

原创

直接用 INCR+EXPIRE 会出错,因为二者非原子操作,中间可能被中断或并发覆盖,导致漏限流;正确做法是用 Lua 脚本封装“查旧值、判超限、更新时间戳、写新值”四步,通过 EVAL/EVALSHA 保证原子性。

redis如何使用lua脚本实现滑动窗口限流

为什么直接用 INCR + EXPIRE 做滑动窗口会出错

因为 Redis 的原子性只保证单命令,INCREXPIRE 是两个独立操作,中间可能被中断或并发覆盖。比如窗口时间刚过期、新请求进来时,旧 key 还没删干净,新计数又从 0 开始,导致漏限流。滑动窗口本质要“查旧值、算是否超限、更新时间戳、写新值”,四步必须原子执行。

正确做法是把整个逻辑塞进 Lua 脚本里,靠 Redis 的 EVAL 保证原子性。脚本里用 redis.call() 操作数据,所有读写都在服务端一次完成。

  • 别在客户端做 if-else 判断后再发命令——网络延迟+竞态会让结果不可信
  • 别用 SETNX + EXPIRE 模拟——同样非原子,且无法表达“过去 N 秒内总请求数”这个语义
  • 注意 Lua 脚本里不能用 os.time(),要用 redis.call('TIME') 获取服务端时间,避免客户端和服务端时钟不同步

EVAL 脚本里怎么维护滑动窗口的时间槽

滑动窗口不是简单存一个总数,而是按时间切片(比如每秒一个槽),记录每个时间段的请求数,再对最近 W 个槽求和。Lua 脚本里常用 zsethash 实现:前者用时间戳作 score 存请求次数,后者用秒级时间戳作 field 存计数。推荐用 zset,天然支持范围查询和自动剔除过期槽。

关键点是每次请求都要:ZREMRANGEBYSCORE 清理过期槽 → ZSCORE 查当前槽是否存在 → ZADD 写入或自增 → ZRANGEBYSCORE 算窗口内总和 → ZCARD 或求和判断是否超限。

  • 窗口宽度(如 60 秒)和时间精度(如 1 秒/槽)要提前定死,不能在脚本里动态算,否则影响性能
  • zset 时,score 必须是整数时间戳(单位秒),别用毫秒——Redis ZRANGEBYSCORE 对浮点 score 支持弱,且精度冗余
  • 如果 QPS 很高(万级/秒),避免每次 ZRANGEBYSCORE 扫全量;改用 ZCOUNT 统计数量,或预存一个总和 key 配合 HINCRBY 更新

实际部署时 Lua 脚本怎么传、怎么调、怎么防误用

别每次请求都用 EVAL 发一长串脚本——网络开销大,且 Redis 无法缓存编译结果。应该先用 SCRIPT LOAD 把脚本加载进服务端,拿到 SHA1 校验值,之后用 EVALSHA 调用,既快又安全。

X Studio
X Studio

网易云音乐·X Studio

下载

调用时注意传参顺序:EVALSHA <sha> <numkeys> <key1> <key2> ... <arg1> <arg2> ...。其中 keys 必须是真实 Redis key(会被 cluster 路由识别),args 是纯参数(如窗口大小、限流阈值)。

  • key 名建议带业务前缀,比如 rate:login:{uid},避免不同接口共用 key 导致误限流
  • args 里不要传复杂结构,Lua 脚本里不解析 JSON;数字就传数字,布尔就传 1/0
  • 本地开发调试用 EVAL 没问题,上线后必须切到 EVALSHA,否则集群环境下可能因 script cache 不一致报 NOSCRIPT 错误

常见错误现象和对应检查点

限流失效、误触发、脚本超时——八成出在边界条件没处理好。比如窗口刚初始化时没设初始值,或者时间戳计算跨了秒级边界,导致某一秒的计数被漏掉。

典型报错:(error) BUSY Redis is busy running a script. You can only call SCRIPT KILL or SHUTDOWN NOSAVE. 表示脚本卡死(比如 while 循环没出口、或用了耗时命令如 KEYS);(error) ERR Error running script (call to f_...): @user_script:xx: user_script:xx: attempt to compare number with string 是类型混用,Lua 里 redis.call() 返回的都是字符串,得显式 tonumber()

  • 脚本里所有 redis.call() 结果,只要参与数值运算,必须 tonumber() 转换,哪怕看起来是数字
  • redis.call('TIME') 拿到的是 {seconds, microseconds} 表,取 res[1] 就够,别直接当数字用
  • 测试时别只压测单 key,要模拟多用户(不同 key)并发,否则发现不了 key 级别的竞争问题
脚本的健壮性不在行数多,而在每一处 if 分支是否覆盖了 time == nil、count == false、zset 为空这些真实存在的空状态。这些地方漏了,线上就是静默失效。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

458

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

549

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

337

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

84

2025.09.10

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1091

2023.08.02

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

848

2023.08.22

counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

203

2023.11.20

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

108

2023.09.25

Python WebSocket实时通信与异步服务开发实践
Python WebSocket实时通信与异步服务开发实践

本专题聚焦 Python 在实时通信场景中的开发实践,系统讲解 WebSocket 协议原理、长连接管理、消息推送机制以及异步服务架构设计。内容包括客户端与服务端通信实现、连接稳定性优化、消息队列集成及高并发处理策略。通过完整案例,帮助开发者构建高效稳定的实时通信系统,适用于聊天应用、实时数据推送等场景。

2

2026.03.18

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号