redis+lua+固定窗口实现分布式限流

用key的过期时间替代固定窗口的时间戳

lua 复制代码
-- KEYS[1]: 限流的key
-- ARGV[1]: 限流窗口大小(秒)
-- ARGV[2]: 限流阈值

local key = KEYS[1]
local window = tonumber(ARGV[1])
local limit = tonumber(ARGV[2])

-- 尝试获取当前计数
local current = redis.call("GET", key)

if current == false then
    -- key不存在,初始化计数器并设置过期时间
    redis.call("SET", key, 1, "EX", window)
    return 1
else
    -- key存在,检查是否超过限制
    if tonumber(current) < limit then
        redis.call("INCR", key)
        return 1
    else
        return 0
    end
end

java客户端

java 复制代码
public class FixedWindowRateLimiterWithTTL {
    private Jedis jedis;
    private String key;
    private int window; // 窗口大小(秒)
    private int limit;  // 限流阈值
    
    private static final String LUA_SCRIPT =
        "local key = KEYS[1]\n" +
        "local window = tonumber(ARGV[1])\n" +
        "local limit = tonumber(ARGV[2])\n" +
        "local current = redis.call(\"GET\", key)\n" +
        "if current == false then\n" +
        "    redis.call(\"SET\", key, 1, \"EX\", window)\n" +
        "    return 1\n" +
        "else\n" +
        "    if tonumber(current) < limit then\n" +
        "        redis.call(\"INCR\", key)\n" +
        "        return 1\n" +
        "    else\n" +
        "        return 0\n" +
        "    end\n" +
        "end";
    
    public FixedWindowRateLimiterWithTTL(Jedis jedis, String key, int window, int limit) {
        this.jedis = jedis;
        this.key = key;
        this.window = window;
        this.limit = limit;
    }
    
    public boolean allowRequest() {
        Object result = jedis.eval(LUA_SCRIPT, 
            Collections.singletonList(key),
            Arrays.asList(
                String.valueOf(window),
                String.valueOf(limit)
            ));
        return "1".equals(result.toString());
    }
}

使用

java 复制代码
public static void main(String[] args) {
    Jedis jedis = new Jedis("localhost");
    // 创建一个每分钟最多100次请求的限流器
    FixedWindowRateLimiterWithTTL limiter = 
        new FixedWindowRateLimiterWithTTL(jedis, "api:limit:user1", 60, 100);
    
    for (int i = 0; i < 120; i++) {
        if (limiter.allowRequest()) {
            System.out.println("处理请求 " + i);
        } else {
            System.out.println("限流请求 " + i);
        }
    }
    
    jedis.close();
}

优点:实现简单

缺点:

固定窗口算法无法解决临界问题

Redis的过期机制是惰性删除+定期删除,可能导致key实际过期时间与预期有微小差异

重启导致的窗口重置
在超高并发下会成为单点瓶颈

相关推荐
曲幽24 分钟前
我用了FastApiAdmin后,连夜把踩过的坑都整理出来了
redis·python·postgresql·vue3·fastapi·web·sqlalchemy·admin·fastapiadmin
元拓数智2 小时前
智能分析落地卡壳?先补好「数据关系+语义治理」这层技术基建
大数据·分布式·ai·spark·数据关系·语义治理
jran-6 小时前
Redis 命令
数据库·redis·缓存
GIS数据转换器6 小时前
农村生活污水治理智慧管控平台
大数据·人工智能·分布式·数据分析·生活·智慧城市
June`7 小时前
多线程redis下如何解决aof重写和rdb持久化的数据一致性问题
数据库·redis·缓存
Trouvaille ~8 小时前
【Redis篇】初识 Redis:特性、应用场景与版本演进
数据结构·数据库·redis·分布式·缓存·中间件·持久化
米高梅狮子10 小时前
Ceph 分布式存储 部署
linux·运维·数据库·分布式·ceph·docker·华为云
郭龙_Jack10 小时前
跨境电商 平台 - ERP - 内部子系统 交互方式总图
分布式·教育电商
喝醉酒的小白10 小时前
Kafka 集群应急故障排查手册
分布式·kafka
无籽西瓜a10 小时前
【西瓜带你学Kafka | 第八期】 Kafka的主从同步、消息可靠性、流处理与顺序消费(文含图解)
java·分布式·后端·kafka·消息队列·mq