java 接口限制访问次数

1. 通过注解的形式实现通用接口访问次数限制

(1)pom.xml文件中引入expiringmap

 <dependency>
      <groupId>net.jodah</groupId>
      <artifactId>expiringmap</artifactId>
      <version>0.5.10</version>
 </dependency>

(2)添加注解

复制代码
@Documented
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface LimitRequest {
    //请求时间
    long time() default 60 * 1000;
    //请求次数
    int count() default Integer.MAX_VALUE;
}
复制代码

(3)添加注解对应解析类

复制代码
@Aspect
@Component
public class LimitRequestAspect {
    private static ConcurrentHashMap<String, ExpiringMap<String, Integer>> hashMap = new ConcurrentHashMap<>();

    @Pointcut("@annotation(limitRequest)")
    public void excudeService(LimitRequest limitRequest) {
    }

    @Around("excudeService(limitRequest)")
    public Object doAround(ProceedingJoinPoint pjp, LimitRequest limitRequest) throws Throwable {

        RequestAttributes ra = RequestContextHolder.getRequestAttributes();
        ServletRequestAttributes sra = (ServletRequestAttributes) ra;
        HttpServletRequest request = sra.getRequest();

        ExpiringMap<String, Integer> map = hashMap.getOrDefault(request.getRequestURI(), ExpiringMap.builder().variableExpiration().build());
        Integer count = map.getOrDefault(request.getRemoteAddr(), 0);

        if (count >= limitRequest.count()) {
            throw new BusinessException(500, "访问次数超过限制");
        } else if (count == 0) {
            map.put(request.getRemoteAddr(), count + 1, ExpirationPolicy.CREATED, limitRequest.time(), TimeUnit.MILLISECONDS);
        } else {
            map.put(request.getRemoteAddr(), count + 1);
        }
        hashMap.put(request.getRequestURI(), map);
        Object result = pjp.proceed();

        return result;
    }
}
复制代码

然后就可以在接口上添加对应的注解@LimitRequest(time=60*1000, count = 3) 来限制时间范围内请求次数了。

2. 针对特殊接口

思路:通过redis 失效时间来控制时间范围内请求次数

String commentKey = String.format("comment:%d:%d", comment.getBusinessId(), comment.getUserId());
Long count = redisTemplate.opsForValue().increment(commentKey);
if (count > 10) {
     throw new BusinessException("发表评论过于频繁,休息下再操作吧~");
   }
redisTemplate.expire(commentKey, 30, TimeUnit.SECONDS);

 

posted @   山阴路的秋天  阅读(3026)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 单线程的Redis速度为什么快?
· SQL Server 2025 AI相关能力初探
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
· 展开说说关于C#中ORM框架的用法!
点击右上角即可分享
微信分享提示