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

使用令牌桶算法通过redis实现限流

令牌桶算法是一种常用的限流算法,它可以平滑地控制请求的处理速率。在 Java 中结合 Redis 实现令牌桶算法,可以利用 Redis 的原子操作来保证多节点环境下的限流效果。

一 实现思路

  1. 初始化令牌桶:在 Redis 中存储令牌桶的相关信息,如令牌数量、令牌桶容量、上一次更新时间等。
  2. 添加令牌:根据时间间隔,计算需要添加到令牌桶中的令牌数量,并更新令牌桶的令牌数量。
  3. 获取令牌:尝试从令牌桶中获取指定数量的令牌,如果令牌数量足够,则返回成功;否则返回失败。

二 Redis Lua 脚本

编写一个 Lua 脚本来处理令牌桶的逻辑。lua脚本会在 Redis 执行时保证原子性。

-- 最大令牌个数(qps)
local limit = tonumber(ARGV[1]) 
-- 令牌生成速率
local rate = tonumber(ARGV[2]) 
-- 当前时间秒*1000=毫秒
local now = redis.call('TIME')[1] * 1000
-- 剩余可用的令牌个数
local tokens = tonumber(redis.call('GET', KEYS[1]) or 0)
-- 上次访问时间戳(单位毫秒)
local lastTimestamp = tonumber(redis.call('GET', KEYS[2]) or 0)
-- 本次访问时间戳-上次访问时间戳
local elapsedTime = now - lastTimestamp
-- 计算本次访问与上次访问之间需要放入的令牌个数=上次剩余的令牌数+本次访问与上次访问之间差的令牌数
local newTokens = tokens + elapsedTime * rate / 1000
-- 如果需要放入的令牌个数超过限制的最大令牌个数,则直接使用最大令牌个数
if newTokens > limit then
	newTokens = limit 
end 
-- 如果需要放入的令牌个数小于1,则意味着无法获取到锁
if newTokens < 1 then
	return 0 
else
-- 获取到锁,则更新剩余可获取的锁个数-1,更新本次访问时间戳
	redis.call('SET', KEYS[1], newTokens - 1)
	redis.call('SET', KEYS[2], now)
	return 1 
end

三 Java 实现

public class RedisRateLimiter {

  private JedisPool jedisPool;

  private String LUA_SCRIPT = "上面的lua脚本替换到此处";

  public RedisRateLimiter(JedisPool jedisPool) {
    this.jedisPool = jedisPool;
  }

  public boolean tryAcquire(String key, int rate, int capacity) {
        try (Jedis jedis = jedisPool.getResource()) {
      Object result = jedis.eval(LUA_SCRIPT,
          Arrays.asList(key, key + ":timestamp"), Arrays.asList(capacity, rate));
      long response = (Long) result;
      return response != 0;
    } catch (Exception e) {
        return false;
    }
  }
}


http://www.kler.cn/a/537739.html

相关文章:

  • R语言 文本分析 天龙八部
  • 论文阅读:MGMAE : Motion Guided Masking for Video Masked Autoencoding
  • 离线安装Appium Server
  • pandas 读取excel数据,以字典形式输出
  • 【Spring Boot】 SpringBoot自动装配-Condition
  • OSPF基础(2):数据包详解
  • WP.29-R155 汽车网络安全法规解读
  • 38、【OS】【Nuttx】OSTest分析(3):参数传递
  • 云计算能加速产品的上市时间吗?
  • Python中3个与众不同的运算符 :=海象 ->箭头 //地板除法运算符
  • 力扣LeetCode: 63 不同路径Ⅱ
  • AtCoder Beginner Contest 392(A-G)题解
  • 用AVFrame + AVPacket 完成accede编码和直接用ffmpeg命令行实现acc编码的对比
  • 肖特基二极管正向浪涌电流,如何左右电路命运?
  • 编程中的科学计数法
  • 使用C++实现多个经典算法
  • mysql8安装时提示-缺少Microsoft Visual C++ 2019 x64 redistributable
  • 【场景题】架构优化 - 解耦Redis缓存与业务逻辑
  • LLM:DeepSeek 系列(三)
  • 【代码随想录|子序列系列300,674,718】
  • Qwen2-VL:增强视觉语言模型对世界任意分辨率的感知能力
  • 如何使用DeepSeek
  • mybatis 是否支持延迟加载?延迟加载的原理是什么?
  • systemverilog的program和module的区别
  • ReentrantLock/内存溢出监控
  • 南京观海微电子----开关电源电路图