自定义注解+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

相关文章

Spring Boot整合Redis注解实现增删改查功能(Redis注解使用)

《SpringBoot整合Redis注解实现增删改查功能(Redis注解使用)》文章介绍了如何使用SpringBoot整合Redis注解实现增删改查功能,包括配置、实体类、Repository、Se... 目录配置Redis连接定义实体类创建Repository接口增删改查操作示例插入数据查询数据删除数据更

Java Lettuce 客户端入门到生产的实现步骤

《JavaLettuce客户端入门到生产的实现步骤》本文主要介绍了JavaLettuce客户端入门到生产的实现步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要... 目录1 安装依赖MavenGradle2 最小化连接示例3 核心特性速览4 生产环境配置建议5 常见问题

linux ssh如何实现增加访问端口

《linuxssh如何实现增加访问端口》Linux中SSH默认使用22端口,为了增强安全性或满足特定需求,可以通过修改SSH配置来增加或更改SSH访问端口,具体步骤包括修改SSH配置文件、增加或修改... 目录1. 修改 SSH 配置文件2. 增加或修改端口3. 保存并退出编辑器4. 更新防火墙规则使用uf

Java 的ArrayList集合底层实现与最佳实践

《Java的ArrayList集合底层实现与最佳实践》本文主要介绍了Java的ArrayList集合类的核心概念、底层实现、关键成员变量、初始化机制、容量演变、扩容机制、性能分析、核心方法源码解析、... 目录1. 核心概念与底层实现1.1 ArrayList 的本质1.1.1 底层数据结构JDK 1.7

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

Python实现字典转字符串的五种方法

《Python实现字典转字符串的五种方法》本文介绍了在Python中如何将字典数据结构转换为字符串格式的多种方法,首先可以通过内置的str()函数进行简单转换;其次利用ison.dumps()函数能够... 目录1、使用json模块的dumps方法:2、使用str方法:3、使用循环和字符串拼接:4、使用字符