lua脚本+redis实现限流
创始人
2025-05-29 05:09:32
0

定义注解 @RateLimiter

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface RateLimiter 
{// 限流keypublic String key() default Constants.RATE_LIMIT_KEY;// 限流时间,单位秒public int time() default 60;// 限流次数public int count() default 100;// 限流类型public LimitType limitType() default LimitType.DEFAULT;
}public enum LimitType
{/*** 默认策略全局限流*/DEFAULT,/*** 根据请求者IP进行限流*/IP
}

一个作用在方法上的注解,有四个属性

key:存储在redis里用到的key
time:限流时间,相当于redis里的有效期
count:限流次数
limitType: 限流类型,点开枚举发现有默认和IP两种限流方式,这两种方式的实现只是存储在redis里的key不同
2. 切面
我们来看一看@RateLimiter这个注解的切面RateLimiterAspect.java

@Aspect
@Component
public class RateLimiterAspect 
{private static final Logger log = LoggerFactory.getLogger(RateLimiterAspect.class);private RedisTemplate redisTemplate;private RedisScript limitScript;@Autowiredpublic void setRedisTemplate1(RedisTemplate redisTemplate){this.redisTemplate = redisTemplate;}@Autowiredpublic void setLimitScript(RedisScript limitScript){this.limitScript = limitScript;}@Before("@annotation(rateLimiter)")public void doBefore(JoinPoint point, RateLimiter rateLimiter) throws Throwable{String key = rateLimiter.key();int time = rateLimiter.time();int count = rateLimiter.count();String combineKey = getCombineKey(rateLimiter, point);List keys = Collections.singletonList(combineKey);try{// 调用lua脚本,传入三个参数Long number = redisTemplate.execute(limitScript, keys, count, time);if (StringUtils.isNull(number) || number.intValue() > count){throw new ServiceException("访问过于频繁,请稍候再试");}log.info("限制请求'{}',当前请求'{}',缓存key'{}'", count, number.intValue(), key);}catch (ServiceException e){throw e;}catch (Exception e){throw new RuntimeException("服务器限流异常,请稍候再试");}}public String getCombineKey(RateLimiter rateLimiter, JoinPoint point){// 获取注解中的key值StringBuffer stringBuffer = new StringBuffer(rateLimiter.key());// 判断限流类型,如果是IP限流,就在key后添加上IP(若依自己写了一个获取ip的方法类,大家可以自行查看)if (rateLimiter.limitType() == LimitType.IP){stringBuffer.append(IpUtils.getIpAddr(ServletUtils.getRequest())).append("-");}// 获取方法MethodSignature signature = (MethodSignature) point.getSignature();Method method = signature.getMethod();// 获取类Class targetClass = method.getDeclaringClass();// key中添加方法名-类名stringBuffer.append(targetClass.getName()).append("-").append(method.getName());return stringBuffer.toString();}
}
 

简单说明一下这个切面类:

使用了set的方式注入了RedisTemplate和RedisScript,RedisTemplate大家都很熟悉,RedisScript是用于加载和执行lua脚本的
定义了一个前置通知(废话,限流肯定是前置),通过getCombineKey方法获取应该存入redis中的key,getCombineKey方法每一步我都做了注解
将key、time、count作为参数传入lua脚本,执行脚本,判断返回值为空或者或者返回值大于设定的count,抛出异常,由全局异常处理器处理,方法不再往下执行,达到了限流的效果

3.lua脚本
最后,我们来看一看若依是怎么写lua脚本的,在脚本在redis的配置类RedisConfig.java

@Configuration
@EnableCaching
public class RedisConfig extends CachingConfigurerSupport
{……@Beanpublic DefaultRedisScript limitScript(){// 泛型是返回值的类型DefaultRedisScript redisScript = new DefaultRedisScript<>();// 设置脚本redisScript.setScriptText(limitScriptText());// 设置返回值类型redisScript.setResultType(Long.class);return redisScript;}/*** 限流脚本*/private String limitScriptText(){return "local key = KEYS[1]\n" +"local count = tonumber(ARGV[1])\n" +"local time = tonumber(ARGV[2])\n" +"local current = redis.call('get', key);\n" +"if current and tonumber(current) > count then\n" +"    return tonumber(current);\n" +"end\n" +"current = redis.call('incr', key)\n" +"if tonumber(current) == 1 then\n" +"    redis.call('expire', key, time)\n" +"end\n" +"return tonumber(current);";}
}

我们主要看下lua脚本:

接收3个变量:key,阈值count,过期时间time
调用get(key)方法获取key中的值current,如果这个key存在并且current大于count,返回current
调用redis的自增函数赋值给current,当current=1时(即第一次访问该接口),调用redis的设置过期时间函数给当前key设置过期时间
返回current
使用lua脚本可以在并发的情况下更好的满足原子性,只是我不太明白若依为什么不把脚本文件单独拿出来写在resources文件夹下,这样阅读和维护都会更加方便。总之,这就是若依限流注解的全部内容

总结
标注了@RateLimiter注解的方法,在执行方法前调用lua脚本,把自己的类名+方法名当做key传入,判断返回值是否大于设定的阈值,大于则抛出异常不再向下执行,异常由全局异常处理器处理。

相关内容

热门资讯

JavaWeb开发(四)4.2... 一、SpringMVC 请求与响应 1、@RequestMapping 注解 @Re...
开学日记500字 开学日记500字  开学,汉语解释有开设学校、启作学者、学期开始三个释义。以下是小编收集的开学日记相...
朝花夕拾读书笔记摘抄好词好句 朝花夕拾读书笔记摘抄好词好句  一、朝花夕拾的主要内容  《朝花夕拾》里作者鲁迅用夹叙夹议的方法,以...
net.coobird.thu... 1.简单介绍 net.coobird.thumbnailator.Thumbnails 是一个用于创...
《红楼梦》读书笔记 《红楼梦》读书笔记600字《红楼梦》读书笔记600字娴静似娇花照水,行动如弱柳扶风,心较比干多一窍,...
增广贤文读书笔记 增广贤文读书笔记(精选30篇)  读完一本名著以后,想必你有不少可以分享的东西,何不写一篇读书笔记记...
[解决]Splunk KV S... 1: 背景: 今天客户反映数据搜索的时候,不是很稳定,想确定一下 ,这个是什么原因造成的,我去系统里...
爱的教育作文 爱的教育作文爱的教育看到这个书名,我不禁开始思考一个问题:在这个缤纷多彩的世界里,爱究竟是什么含义?...
梁衡《把栏杆拍遍》读书笔记 梁衡《把栏杆拍遍》读书笔记梁衡《把栏杆拍遍》读书笔记这是一篇写得很美的散文,有以下特点:一、联想丰富...
【每日一题Day150】LC1... 分割两个字符串得到回文串【LC1616】 给你两个字符串 a 和 b ,它们长度相同...
一年级春游日记 一年级春游日记一年级春游日记1  今天是春游,我作天就去买许多零食和矿泉水,打算在春游的时候干掉,我...
课外读书笔记摘抄 课外读书笔记摘抄(精选12篇)  导语:舍弃就是这样,它也许出于无奈,可在无奈之后是另一份希望,它也...
蚂蚁观察日记 【热门】蚂蚁观察日记4篇蚂蚁观察日记 篇1  我家有一个后院,我经常到后院去观察那些鹭绿上得小精灵—...
ImageView(图像视图) 本节介绍的UI基础控件是:ImageView(图像视图),就是用来显示图像的一个View或者说控件!...
关于接口测试——自动化框架的设... 一、自动化测试框架 在大部分测试人员眼中只要沾上“框架”,就感觉非常神秘,...
【2023.3.8】数据结构复... 【2023.3.8】数据结构复习笔记 文章目录【2023.3.8】数据结构复习笔记序言一、绪论二、线...
数学周记 【精华】数学周记范文(通用20篇)  不经意间,一个星期已经结束了,想必有很多难忘的瞬间吧,是时候仔...
男生贾里全传读书笔记 男生贾里全传读书笔记(通用24篇)  当品读完一部作品后,相信大家一定领会了不少东西,不能光会读哦,...
1.计算机网络和因特网 1.因特网的描述1.1 具体构成描述根据底层实现(硬件软件)端系统(主机...
使用YOLO部署哨岗相机 流程 一.模型选取 将YOLO和Faster RCNN进行搭配,通过多次实验ÿ...