自定义注解+AOP+SPEL表达式+Redis实现自定义限流注解

2024-03-08 09:12

本文主要是介绍自定义注解+AOP+SPEL表达式+Redis实现自定义限流注解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

自定义注解
/*** 速率限制注解** @author: 张定辉* @date: 2024/3/5 21:29* @description: 速率限制注解*/
@Target({ElementType.TYPE,ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface RateLimit {/*** SPEL表达式* <p>* 1.使用方法的基本类型参数作为限流Key* <p>* &#064;RateLimit(value="#id")* public void test(String id){}* <p><p>* 2.使用方法的对象类型参数中的某个属性作为限流Key* <p>* &#064;RateLimit(value="#user.username")* public void test(User user){}* <p><p>* 3.将方法参数作为bean方法的参数并获取返回值作为限流Key,暂时只支持bean的方法是String类型* <p>* &#064;Service(value="parseBean")<p>* public class ParseBean{<p>* &nbsp;&nbsp;&nbsp;public String parse(String arg){<p>* &nbsp;&nbsp;&nbsp;&nbsp;return arg+"limitKey";<p>* &nbsp;&nbsp;&nbsp;}<p>* }<p>*<p>* &#064;RateLimit(value="@parseBean.parse(username)")<p>* public void test(String username){}*/String value();/*** 限流间隔,以秒为单位*/int interval()default 3;/*** 单位之间内的速率限制*/int frequency()default 20;
}
SPEL配置类
/*** Spel表达式配置类** @author: 张定辉* @date: 2024/3/7 14:20* @description: Spel表达式配置类*/
@Configuration
public class SpelConfig {@Beanpublic StandardEvaluationContext evaluationContext(ApplicationContext applicationContext) {StandardEvaluationContext context = new StandardEvaluationContext();context.addPropertyAccessor(new BeanFactoryAccessor());context.setBeanResolver(new BeanFactoryResolver(applicationContext));context.setTypeLocator(new StandardTypeLocator(applicationContext.getClassLoader()));context.setTypeConverter(new StandardTypeConverter());return context;}
}
AOP切面
/*** 速率限制注解处理器** @author: 张定辉* @date: 2024/3/5 21:37* @description: 速率限制注解处理器*/
@Aspect
@Component
@RequiredArgsConstructor
public class RateLimitHandler {private final ApplicationContext applicationContext;private final SpelExpressionParser parser = new SpelExpressionParser();private final StandardEvaluationContext context;private final RedisTemplate<String, Object> redisTemplate;@SneakyThrows@Around("@within(com.ai.common.annotation.RateLimit) || @annotation(com.ai.common.annotation.RateLimit)")public Object handler(ProceedingJoinPoint joinPoint) {Object target = joinPoint.getTarget();String spelValue;int interval;int frequency;//如果注解是标注在类上if (target.getClass().isAnnotationPresent(RateLimit.class)) {Class<?> aClass = target.getClass();RateLimit annotation = aClass.getAnnotation(RateLimit.class);spelValue = annotation.value();interval = annotation.interval();frequency = annotation.frequency();if (spelValue.startsWith("@")) {addBeanResultToContext(context, spelValue);}}//注解标注在方法上else {Object[] args = joinPoint.getArgs();MethodSignature signature = (MethodSignature) joinPoint.getSignature();Method method = signature.getMethod();RateLimit rateLimit = method.getAnnotation(RateLimit.class);interval = rateLimit.interval();frequency = rateLimit.frequency();spelValue = rateLimit.value();String[] parameterNames = signature.getParameterNames();for (int i = 0; i < args.length; i++) {//这行代码在后续的使用bean的方法返回值作为KEY限流时有用处context.setVariable(parameterNames[i], args[i]);if (args[i] != null && !isPrimitive(args[i].getClass())) {addObjectPropertiesToContext(context, parameterNames[i], args[i]);}}if (spelValue.startsWith("@")) {spelValue = addBeanResultToContext(context, spelValue);}}Expression expression = parser.parseExpression(spelValue);Object key = expression.getValue(context);//使用Redis进行限流redisRateLimit(JSON.toJSONString(key), interval, frequency);return joinPoint.proceed();}/*** 添加对象属性值到SPEL上下文环境中*/@SneakyThrowsprivate void addObjectPropertiesToContext(StandardEvaluationContext context, String paramName, Object arg) {Class<?> clazz = arg.getClass();Method[] methods = clazz.getMethods();for (Method method : methods) {String methodName = method.getName();if (methodName.startsWith("get") && !methodName.equals("getClass")) {String propertyName = methodName.substring(3, 4).toLowerCase() + methodName.substring(4);Object propertyValue = method.invoke(arg);context.setVariable(paramName + "." + propertyName, propertyValue);}}}/*** 将Bean方法的执行结果设置到SPEL上下文环境中*/@SneakyThrowsprivate String addBeanResultToContext(StandardEvaluationContext context, String spelValue) {Object bean = applicationContext.getBean(spelValue.substring(1, spelValue.indexOf(".")));String methodName = spelValue.substring(spelValue.indexOf(".") + 1, spelValue.indexOf("("));String[] methodArgs = spelValue.substring(spelValue.indexOf("(") + 1, spelValue.indexOf(")")).split(",");Object[] methodArgsValues = new Object[methodArgs.length];for (int i = 0; i < methodArgs.length; i++) {methodArgsValues[i] = context.lookupVariable(methodArgs[i]);if (Objects.isNull(methodArgsValues[i])) {methodArgsValues[i] = methodArgs[i];}}Class<?>[] argumentsTypes = getArgumentsTypes(methodArgsValues);boolean b = Arrays.stream(argumentsTypes).allMatch(Objects::isNull);Method beanMethod = bean.getClass().getMethod(methodName, b?new Class<?>[0]:argumentsTypes);Object beanMethodResult = beanMethod.invoke(bean, b?null:methodArgsValues);context.setVariable("beanMethodResult", beanMethodResult);return "#beanMethodResult";}/*** 获取参数的类型*/private Class<?>[] getArgumentsTypes(Object[] args) {Class<?>[] types = new Class<?>[args.length];for (int i = 0; i < args.length; i++) {Class<?> aClass = args[i].getClass();if (aClass.isAssignableFrom(String.class)) {String arg = (String) args[i];types[i] = StringUtils.isBlank(arg) ? null : aClass;} else {types[i] = aClass;}}return types;}/*** 判断是否是基础数据类型*/private boolean isPrimitive(Class<?> clazz) {return clazz.isPrimitive() || clazz == String.class || clazz == Integer.class|| clazz == Long.class || clazz == Double.class || clazz == Float.class|| clazz == Boolean.class || clazz == Character.class || clazz == Short.class|| clazz == Byte.class;}/*** 结合Redis进行限流操作*/private void redisRateLimit(String key, int interval, int frequency) throws OperationsException {long l = execLua(key, interval);if (l > frequency) {throw new OperationsException("操作过于频繁,请稍后再试!");}}/*** 使用Lua脚本执行原子性的Redis操作,* 如果key不存在则设置value为1并且设置过期时间为5秒,* 如果key存在则进行累加。避免多线程并发时,由于key被修改过导致设置过期时间时失败从而导致key永不失效** @return 如果没有key则返回1,如果有key则返回累加后的value*/private long execLua(String key, int expireTime) {String luaScript = """if redis.call('exists', KEYS[1]) == 0 thenredis.call('set', KEYS[1], 1, 'ex', %s)return 1elsereturn redis.call('incr',KEYS[1])end""".formatted(expireTime);RedisScript<Long> script = new DefaultRedisScript<>(luaScript, Long.class);Long result = redisTemplate.execute(script, Collections.singletonList(key));return Objects.isNull(result) ? 0 : result;}
}
定义Bean方法解析的业务类

该业务类主要是为了满足在使用自定义注解时我们会使用某个类的方法的返回值作为限流Key,这个类自己自定义即可,这里只是做简单的演示使用

/*** @author: 张定辉* @date: 2024/3/7 11:48* @description: 使用方法返回值作为限流Key的业务方法*/
@Service(value = "parseService")
public class ParseService {public String parse(String param){return param+"yyds";}public String parse2(){return "yyds";}
}
实际应用
注解标注在接口方法上,使用方法参数作为限流Key

5秒内只能访问两次该接口

  @GetMapping("/test")@RateLimit(value = "#id",interval = 5,frequency = 2)public Res<Object> test(@RequestParam String id){return Res.success();}
标注在接口方法上,使用对象的属性值作为限流Key

5秒内只能访问两次该接口

   @PostMapping("/test")@RateLimit(value = "#user.username",interval = 5,frequency = 2)public Res<Object> test(@RequestBody User user){return Res.success();}
标注在接口方法上,使用 parseService 业务类型的 parse方法返回值作为限流Key

5秒内只能访问两次该接口

   @GetMapping("/test")@RateLimit(value = "@parseService.parse(id)",interval = 5,frequency = 2)public  Res<Object> test(@RequestParam String id){return Res.success();}
标注在接口类下,实现该接口下的所有接口方法都限流
/**
* @author: 张定辉
* @date: 2024/3/7 14:14
* @description:
*/
@RequestMapping("/test")
@RestController
@RateLimit(value = "@parseService.parse2()",interval = 5,frequency = 2)
public class Text2Controller {@GetMapping("/test1")public Res<Object> test1(){return Res.success();}@GetMapping("/test2")public Res<Object> test2(){return Res.success();}
}

写的可能不是很完善,如果有大佬能够指正的话不甚感激

这篇关于自定义注解+AOP+SPEL表达式+Redis实现自定义限流注解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/786638

相关文章

mybatis执行insert返回id实现详解

《mybatis执行insert返回id实现详解》MyBatis插入操作默认返回受影响行数,需通过useGeneratedKeys+keyProperty或selectKey获取主键ID,确保主键为自... 目录 两种方式获取自增 ID:1. ​​useGeneratedKeys+keyProperty(推

Spring Boot集成Druid实现数据源管理与监控的详细步骤

《SpringBoot集成Druid实现数据源管理与监控的详细步骤》本文介绍如何在SpringBoot项目中集成Druid数据库连接池,包括环境搭建、Maven依赖配置、SpringBoot配置文件... 目录1. 引言1.1 环境准备1.2 Druid介绍2. 配置Druid连接池3. 查看Druid监控

Linux在线解压jar包的实现方式

《Linux在线解压jar包的实现方式》:本文主要介绍Linux在线解压jar包的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux在线解压jar包解压 jar包的步骤总结Linux在线解压jar包在 Centos 中解压 jar 包可以使用 u

c++ 类成员变量默认初始值的实现

《c++类成员变量默认初始值的实现》本文主要介绍了c++类成员变量默认初始值,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录C++类成员变量初始化c++类的变量的初始化在C++中,如果使用类成员变量时未给定其初始值,那么它将被

Qt使用QSqlDatabase连接MySQL实现增删改查功能

《Qt使用QSqlDatabase连接MySQL实现增删改查功能》这篇文章主要为大家详细介绍了Qt如何使用QSqlDatabase连接MySQL实现增删改查功能,文中的示例代码讲解详细,感兴趣的小伙伴... 目录一、创建数据表二、连接mysql数据库三、封装成一个完整的轻量级 ORM 风格类3.1 表结构

基于Python实现一个图片拆分工具

《基于Python实现一个图片拆分工具》这篇文章主要为大家详细介绍了如何基于Python实现一个图片拆分工具,可以根据需要的行数和列数进行拆分,感兴趣的小伙伴可以跟随小编一起学习一下... 简单介绍先自己选择输入的图片,默认是输出到项目文件夹中,可以自己选择其他的文件夹,选择需要拆分的行数和列数,可以通过

Python中将嵌套列表扁平化的多种实现方法

《Python中将嵌套列表扁平化的多种实现方法》在Python编程中,我们常常会遇到需要将嵌套列表(即列表中包含列表)转换为一个一维的扁平列表的需求,本文将给大家介绍了多种实现这一目标的方法,需要的朋... 目录python中将嵌套列表扁平化的方法技术背景实现步骤1. 使用嵌套列表推导式2. 使用itert

Python使用pip工具实现包自动更新的多种方法

《Python使用pip工具实现包自动更新的多种方法》本文深入探讨了使用Python的pip工具实现包自动更新的各种方法和技术,我们将从基础概念开始,逐步介绍手动更新方法、自动化脚本编写、结合CI/C... 目录1. 背景介绍1.1 目的和范围1.2 预期读者1.3 文档结构概述1.4 术语表1.4.1 核

在Linux中改变echo输出颜色的实现方法

《在Linux中改变echo输出颜色的实现方法》在Linux系统的命令行环境下,为了使输出信息更加清晰、突出,便于用户快速识别和区分不同类型的信息,常常需要改变echo命令的输出颜色,所以本文给大家介... 目python录在linux中改变echo输出颜色的方法技术背景实现步骤使用ANSI转义码使用tpu

Knife4j+Axios+Redis前后端分离架构下的 API 管理与会话方案(最新推荐)

《Knife4j+Axios+Redis前后端分离架构下的API管理与会话方案(最新推荐)》本文主要介绍了Swagger与Knife4j的配置要点、前后端对接方法以及分布式Session实现原理,... 目录一、Swagger 与 Knife4j 的深度理解及配置要点Knife4j 配置关键要点1.Spri