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

相关文章

Redis客户端连接机制的实现方案

《Redis客户端连接机制的实现方案》本文主要介绍了Redis客户端连接机制的实现方案,包括事件驱动模型、非阻塞I/O处理、连接池应用及配置优化,具有一定的参考价值,感兴趣的可以了解一下... 目录1. Redis连接模型概述2. 连接建立过程详解2.1 连php接初始化流程2.2 关键配置参数3. 最大连

Python实现网格交易策略的过程

《Python实现网格交易策略的过程》本文讲解Python网格交易策略,利用ccxt获取加密货币数据及backtrader回测,通过设定网格节点,低买高卖获利,适合震荡行情,下面跟我一起看看我们的第一... 网格交易是一种经典的量化交易策略,其核心思想是在价格上下预设多个“网格”,当价格触发特定网格时执行买

python设置环境变量路径实现过程

《python设置环境变量路径实现过程》本文介绍设置Python路径的多种方法:临时设置(Windows用`set`,Linux/macOS用`export`)、永久设置(系统属性或shell配置文件... 目录设置python路径的方法临时设置环境变量(适用于当前会话)永久设置环境变量(Windows系统

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

Python对接支付宝支付之使用AliPay实现的详细操作指南

《Python对接支付宝支付之使用AliPay实现的详细操作指南》支付宝没有提供PythonSDK,但是强大的github就有提供python-alipay-sdk,封装里很多复杂操作,使用这个我们就... 目录一、引言二、准备工作2.1 支付宝开放平台入驻与应用创建2.2 密钥生成与配置2.3 安装ali

Spring Security 单点登录与自动登录机制的实现原理

《SpringSecurity单点登录与自动登录机制的实现原理》本文探讨SpringSecurity实现单点登录(SSO)与自动登录机制,涵盖JWT跨系统认证、RememberMe持久化Token... 目录一、核心概念解析1.1 单点登录(SSO)1.2 自动登录(Remember Me)二、代码分析三、

PyCharm中配置PyQt的实现步骤

《PyCharm中配置PyQt的实现步骤》PyCharm是JetBrains推出的一款强大的PythonIDE,结合PyQt可以进行pythion高效开发桌面GUI应用程序,本文就来介绍一下PyCha... 目录1. 安装China编程PyQt1.PyQt 核心组件2. 基础 PyQt 应用程序结构3. 使用 Q

springboot自定义注解RateLimiter限流注解技术文档详解

《springboot自定义注解RateLimiter限流注解技术文档详解》文章介绍了限流技术的概念、作用及实现方式,通过SpringAOP拦截方法、缓存存储计数器,结合注解、枚举、异常类等核心组件,... 目录什么是限流系统架构核心组件详解1. 限流注解 (@RateLimiter)2. 限流类型枚举 (

Redis MCP 安装与配置指南

《RedisMCP安装与配置指南》本文将详细介绍如何安装和配置RedisMCP,包括快速启动、源码安装、Docker安装、以及相关的配置参数和环境变量设置,感兴趣的朋友一起看看吧... 目录一、Redis MCP 简介二、安www.chinasem.cn装 Redis MCP 服务2.1 快速启动(推荐)2.

Python实现批量提取BLF文件时间戳

《Python实现批量提取BLF文件时间戳》BLF(BinaryLoggingFormat)作为Vector公司推出的CAN总线数据记录格式,被广泛用于存储车辆通信数据,本文将使用Python轻松提取... 目录一、为什么需要批量处理 BLF 文件二、核心代码解析:从文件遍历到数据导出1. 环境准备与依赖库