0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

Redis实现限流的三种方式分享

马哥Linux运维 来源:CSDN 2023-02-22 09:52 次阅读

面对越来越多的高并发场景,限流显示的尤为重要。

当然,限流有许多种实现的方式,Redis具有很强大的功能,我用Redis实践了三种的实现方式,可以较为简单的实现其方式。

Redis不仅仅是可以做限流,还可以做数据统计,附近的人等功能,这些可能会后续写到。

第一种:基于Redis的setnx的操作

我们在使用Redis的分布式锁的时候,大家都知道是依靠了setnx的指令,在CAS(Compare and swap)的操作的时候,同时给指定的key设置了过期时间(expire),我们在限流的主要目的就是为了在单位时间内,有且仅有N数量的请求能够访问我的代码程序。所以依靠setnx可以很轻松的做到这方面的功能。

比如我们需要在10秒内限定20个请求,那么我们在setnx的时候可以设置过期时间10,当请求的setnx数量达到20时候即达到了限流效果。代码比较简单就不做展示了。

当然这种做法的弊端是很多的,比如当统计1-10秒的时候,无法统计2-11秒之内,如果需要统计N秒内的M个请求,那么我们的Redis中需要保持N个key等等问题

第二种:基于Redis的数据结构zset

其实限流涉及的最主要的就是滑动窗口,上面也提到1-10怎么变成2-11。其实也就是起始值和末端值都各+1即可。

而我们如果用Redis的list数据结构可以轻而易举的实现该功能

我们可以将请求打造成一个zset数组,当每一次请求进来的时候,value保持唯一,可以用UUID生成,而score可以用当前时间戳表示,因为score我们可以用来计算当前时间戳之内有多少的请求数量。而zset数据结构也提供了range方法让我们可以很轻易的获取到2个时间戳内有多少请求

publicResponselimitFlow(){
LongcurrentTime=newDate().getTime();
System.out.println(currentTime);
if(redisTemplate.hasKey("limit")){
Integercount=redisTemplate.opsForZSet().rangeByScore("limit",currentTime-intervalTime,currentTime).size();//intervalTime是限流的时间
System.out.println(count);
if(count!=null&&count>5){
returnResponse.ok("每分钟最多只能访问5次");
}
}
redisTemplate.opsForZSet().add("limit",UUID.randomUUID().toString(),currentTime);
returnResponse.ok("访问成功");
}

通过上述代码可以做到滑动窗口的效果,并且能保证每N秒内至多M个请求,缺点就是zset的数据结构会越来越大。实现方式相对也是比较简单的。

第三种:基于Redis的令牌桶算法

提到限流就不得不提到令牌桶算法了。具体可以参照度娘的解释 令牌桶算法

令牌桶算法提及到输入速率和输出速率,当输出速率大于输入速率,那么就是超出流量限制了。

也就是说我们每访问一次请求的时候,可以从Redis中获取一个令牌,如果拿到令牌了,那就说明没超出限制,而如果拿不到,则结果相反。

依靠上述的思想,我们可以结合Redis的List数据结构很轻易的做到这样的代码,只是简单实现

依靠List的leftPop来获取令牌

//输出令牌
publicResponselimitFlow2(Longid){
Objectresult=redisTemplate.opsForList().leftPop("limit_list");
if(result==null){
returnResponse.ok("当前令牌桶中无令牌");
}
returnResponse.ok(articleDescription2);
}

再依靠Java的定时任务,定时往List中rightPush令牌,当然令牌也需要唯一性,所以我这里还是用UUID进行了生成

//10S的速率往令牌桶中添加UUID,只为保证唯一性
@Scheduled(fixedDelay=10_000,initialDelay=0)
publicvoidsetIntervalTimeTask(){
redisTemplate.opsForList().rightPush("limit_list",UUID.randomUUID().toString());
}

综上,代码实现起始都不是很难,针对这些限流方式我们可以在AOP或者filter中加入以上代码,用来做到接口的限流,最终保护你的网站。

Redis其实还有很多其他的用处,他的作用不仅仅是缓存,分布式锁的作用。他的数据结构也不仅仅是只有String,Hash,List,Set,Zset。有兴趣的可以后续了解下他的GeoHash算法;BitMap,HLL以及布隆过滤器数据(Redis4.0之后加入,可以用Docker直接安装redislabs/rebloom)结构。






审核编辑:刘清

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 过滤器
    +关注

    关注

    1

    文章

    417

    浏览量

    19216
  • CAS
    CAS
    +关注

    关注

    0

    文章

    34

    浏览量

    15099
  • UUID
    +关注

    关注

    0

    文章

    22

    浏览量

    8056
  • Redis
    +关注

    关注

    0

    文章

    366

    浏览量

    10555

原文标题:Redis 实现限流的三种方式

文章出处:【微信号:magedu-Linux,微信公众号:马哥Linux运维】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    三种复位方式比较

    三种复位方式比较
    发表于 08-16 17:31

    步进电机的三种驱动方式

    步进电机的三种驱动方式
    发表于 01-12 17:03

    请问以下三种实现C6474多核编程方式哪一最好?为什么?

    三种:1、在其中两个DSP中实现编解码功能(编解码代码相同),另一个DSP实现混音功能;2、用每个DSP的个核分别完成编码、解码和混音;3、把9个核当成9个CPU,每个核完成编、解码
    发表于 06-25 05:17

    请问stm32启动的三种方式是什么意思?

    请群主详细解释下这三种启动方式,看了参考资料不是很明白其意!谢谢!
    发表于 07-17 04:35

    浅析redis的启动方式

    redis三种启动方式
    发表于 10-12 13:47

    启动Redis三种方法

    Redis笔记(1)——安装、卸载、三种方法启动RedisRedis命令使用(干货十足),Redis两种方法设置密码,时间复杂度(更完善哦
    发表于 06-08 16:09

    伺服电机的三种控制方式怎么选

    一般伺服都有三种控制方式:速度控制方式,转矩控制方式,位置控制方式。大多数人想知道的就是这三种
    发表于 01-29 07:28

    常见的三种无线接入方式是什么?

    蓝牙无线组网的优点是什么?常见的三种无线接入方式是什么?蓝牙无线组网原理与上网方案分享
    发表于 05-26 06:33

    STM32实现复位的三种常用方式问题

    STM32实现复位的三种常用方式问题:最近做项目碰到单片机程序运行过程中,需要根据外部输入信号,对单片机进行复位的操作。解决:目前常用的复位方式有硬件复位和两
    发表于 08-12 07:23

    单片机流水灯的三种实现方式

    单片机流水灯的三种实现方式(由上至下,再重下到上)原理图一般使用单片机的P1口 输出,不需要加上拉电阻:由原理图可以看出这里采用低电平输出。一、采用数组方式进行字节定义,最好理解的
    发表于 11-18 07:23

    STM32的三种开发方式分享

    STM32的三种开发方式通常新手在入门STM32的时候,首先都要先选择一要用的开发方式,不同的开发方式会导致你编程的架构是完全不一样的。一
    发表于 12-01 07:59

    STM32三种启动方式是什么

    STM32三种启动方式是什么
    发表于 12-15 07:16

    Redis 分布式锁的正确实现方式

    分布式锁一般有三种实现方式:1. 数据库乐观锁;2. 基于Redis的分布式锁;3. 基于ZooKeeper的分布式锁。
    的头像 发表于 05-31 14:19 3398次阅读

    一文详解限流算法的实现方式

    不依赖外部库的情况下,限流算法有什么实现的思路?本文介绍了3种实现限流方式
    的头像 发表于 05-25 12:00 1263次阅读

    Redis实现分布式多规则限流方式介绍

    市面上很多介绍 Redis 如何实现限流的,但是大部分都有一个缺点,就是只能实现单一的限流,比如 1 分钟访问 1 次或者 60 分钟访问
    的头像 发表于 02-26 10:07 242次阅读
    <b class='flag-5'>Redis</b><b class='flag-5'>实现</b>分布式多规则<b class='flag-5'>限流</b>的<b class='flag-5'>方式</b>介绍