Guava RateLimiter + AOP注解实现单机限流
依赖:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-aop</artifactId> </dependency> <dependency> <groupId>com.google.guava</groupId> <artifactId>guava</artifactId> <version>30.1.1-jre</version> </dependency>
自定义运行时注解:
import java.lang.annotation.*; import java.util.concurrent.TimeUnit; @Target(ElementType.METHOD) @Retention(RetentionPolicy.RUNTIME) @Documented public @interface XcRateLimit { /** * @return */ String value() default ""; /** * 每秒向桶中放入令牌的数量 默认最大即不做限流 * * @return */ double perSecond() default Double.MAX_VALUE; /** * 获取令牌的等待时间 默认0 * * @return */ int timeOut() default 0; /** * 超时时间单位 * * @return */ TimeUnit timeOutUnit() default TimeUnit.MILLISECONDS; }
aop切面进行环绕通知:
import com.google.common.util.concurrent.RateLimiter; import com.xc.xcspringboot.common.annotation.XcRateLimit; import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.Signature; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Pointcut; import org.aspectj.lang.reflect.MethodSignature; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.stereotype.Component; import org.springframework.web.bind.annotation.ResponseBody; import java.lang.reflect.Method; @Aspect @Component public class XcRateLimitAspect { private final static Logger logger = LoggerFactory.getLogger(XcRateLimitAspect.class); private RateLimiter rateLimiter = RateLimiter.create(Double.MAX_VALUE); /** * 定义切点 * 1、通过扫包切入 * 2、带有指定注解切入 */ // @Pointcut("execution(public * com.xc.xcspringboot.*.*(..))") @Pointcut("@annotation(com.xc.xcspringboot.common.annotation.XcRateLimit)") public void checkPointcut() { } @ResponseBody @Around(value = "checkPointcut()") public Object aroundNotice(ProceedingJoinPoint pjp) throws Throwable { logger.info("拦截到了{}方法...", pjp.getSignature().getName()); Signature signature = pjp.getSignature(); MethodSignature methodSignature = (MethodSignature) signature; //获取目标方法 Method targetMethod = methodSignature.getMethod(); if (targetMethod.isAnnotationPresent(XcRateLimit.class)) { //获取目标方法的@LxRateLimit注解 XcRateLimit lxRateLimit = targetMethod.getAnnotation(XcRateLimit.class); rateLimiter.setRate(lxRateLimit.perSecond()); if (!rateLimiter.tryAcquire(lxRateLimit.timeOut(), lxRateLimit.timeOutUnit())) return "同时调用次数过多,请稍后再试!"; } return pjp.proceed(); } }
在controller中使用自定义注解:
@RestController @RequestMapping("/testRateLimiter") @Slf4j public class TestRateLimiterController { @RequestMapping("/testAnnotation") @XcRateLimit(perSecond = 1.0, timeOut = 500) public String testAnnotation() { return "get success"; } }
参考: https://www.cnblogs.com/myseries/p/12634557.html
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 单元测试从入门到精通
· 上周热点回顾(3.3-3.9)
· winform 绘制太阳,地球,月球 运作规律
2020-07-15 Shell排序
2019-07-15 Spring Cloud Hystrix Dashboard的使用 5.1.3
2019-07-15 Spring Cloud Hystrix 服务容错保护 5.1