Python中用Redis的ZSET实现滑动窗口限流,通过ZREMRANGEBYSCORE清理过期请求、ZADD插入新请求、ZCARD统计数量,并用Lua脚本保证原子性;支持按用户/IP/接口等多维度key设计,辅以降级与监控策略。

Python应用中用Redis实现滑动窗口限流,核心是利用Redis的ZSET(有序集合)按时间戳排序存储请求记录,并通过ZREMRANGEBYSCORE自动清理过期项,再用ZCARD统计当前窗口内请求数。不依赖外部库也能高效完成,关键是设计好时间粒度和key结构。
ZSET的score设为毫秒级时间戳,member可设为唯一标识(如用户ID+时间戳或随机UUID),这样既能去重又能按时间排序。每次请求到来时:
ZREMRANGEBYSCORE key -inf (current_timestamp - window_ms)剔除窗口外的旧记录ZADD key current_timestamp member插入新请求ZCARD key获取当前窗口内请求数,与阈值比对例如:1分钟内最多100次请求,window_ms = 60000,所有score小于time.time()*1000 - 60000的成员都会被自动清理。
直接用全局key会成为瓶颈,应按限流维度构造key,比如:
立即学习“Python免费学习笔记(深入)”;
f"rate_limit:uid:{user_id}"
f"rate_limit:ip:{ip_address}"
f"rate_limit:api:{endpoint}:uid:{user_id}"
注意key生命周期无需手动设置过期——只要窗口内无新请求,ZSET自然变空;也可给key加个较长的EXPIRE(如1小时),防止极端情况下残留空key堆积。
上面三步若拆成多个Redis命令,在高并发下可能因执行间隙导致超限(如A刚删完旧数据、B插入前A又插入一次)。推荐用Lua脚本一次性完成:
local key = KEYS[1]
local now = tonumber(ARGV[1])
local window_ms = tonumber(ARGV[2])
local max_count = tonumber(ARGV[3])
<p>redis.call('zremrangebyscore', key, '-inf', '('..(now - window_ms))
local count = redis.call('zcard', key)
if count < max_count then
redis.call('zadd', key, now, ARGV[4])
return 1
else
return 0
endPython中调用:redis.eval(lua_script, 1, key, now_ms, window_ms, max_count, member_id),返回1表示放行,0表示拒绝。
生产环境别让限流失败直接抛异常:
threading.local或LRU cache),仅限临时兜底ZCARD结果,观察分布是否倾斜INCR计数,便于告警(如5分钟内超1000次触发通知)基本上就这些。滑动窗口比固定窗口更平滑,比漏桶/令牌桶更易理解,用好ZSET和Lua,就能在Python服务里稳稳扛住突发流量。
以上就是Python应用如何实现基于Redis的滑动窗口限流逻辑【技巧】的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号