用Redis的Zset数据结构做接口限流

在公司业务项目中遇到了一个高并发接口,因为那个接口对数据库有很高的写和读的需求,再加之公司服务器会跑很多项目,稍不注意就会使服务器宕机,所以想出了接口限流的方法,是项目正常流转. 下面就是redis用set做的一个一分钟限制5次的限流处理.

 

//可以灵活设置时间得接口请求数
@GetMapping("/Sliding")
@ApiOperation("滑动限流")
public String testSlidingWindow() {
    Long currentTime = System.currentTimeMillis();

    if (redisTemplate.hasKey("limit")) {
        // intervalTime是限流的时间
        Long intervalTime = 60000L;
        Integer count = redisTemplate.opsForZSet().rangeByScore("limit", currentTime - intervalTime, currentTime).size();
        System.out.println(count);
        if (count != null && count > 5) {
            System.out.println("静止访问");
            return "每分钟最多只能访问5次";
        }
    }
    redisTemplate.opsForZSet().add("limit", UUID.randomUUID().toString(), currentTime);
    System.out.println("访问成功");
    return "访问成功";
}

 下面是用swagger测试之后的图片

但是后续我们发现这样会使redis中zset的数据结构变大,但是也不影响使用,最好的方法还可以用Redis集成的Ression令牌桶去限流 这个方法

@Resource
private Redisson redisson;

@GetMapping("/Token")
@ApiOperation("桶限流")
public String testTokenBucket() {
    RRateLimiter rateLimiter = redisson.getRateLimiter("myRateLimiter");

    // 最大流速 = 每10秒钟产生1个令牌  10秒一个令牌

    rateLimiter.trySetRate(RateType.OVERALL, 1, 10, RateIntervalUnit.SECONDS);
    //需要1个令牌
    if (rateLimiter.tryAcquire(1)) {
        return "可以访问";
    }
    System.err.println("静止访问");
    return "不好意思,请过十秒钟再来~~~~~~~";

}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值