Java限流器使用

本文介绍了如何在Spring Boot应用中使用自定义注解@RateLimit实现请求速率限制,包括创建令牌桶、处理permitsPerSecond、timeout和timeUnit参数,并通过AspectJ实现AOP环绕通知来拦截并控制访问频率。
摘要由CSDN通过智能技术生成
import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Inherited;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import java.util.concurrent.TimeUnit;


@Inherited
@Documented
@Target({ElementType.TYPE,ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface RateLimit {

    double permitsPerSecond() default 10;

    long timeout() default 0;

    TimeUnit timeUnit() default TimeUnit.MILLISECONDS;
}
import com.google.common.collect.Maps;
import com.google.common.util.concurrent.RateLimiter;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.TimeUnit;
import javax.servlet.http.HttpServletRequest;
import lombok.extern.slf4j.Slf4j;
import org.apache.ibatis.annotations.Results;
import org.aspectj.lang.ProceedingJoinPoint;
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.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;


@Scope
@Aspect
@Component
@Slf4j
public class RateLimitAspect {

    private Map<String, RateLimiter> limitMap = Maps.newConcurrentMap();

    @Pointcut("@annotation(com.xxx.RateLimit)")
    private void pointcut() {
    }

    @Around(value = "pointcut()")
    public Object around(ProceedingJoinPoint joinPoint) {
        Object obj = null;
        try {

            RateLimit rateLimit = ((MethodSignature) joinPoint.getSignature()).getMethod()
                    .getAnnotation(RateLimit.class);

            HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes())
                    .getRequest();
            String url = request.getRequestURI();
            if (rateLimit != null) {
                // 获取注解的permitsPerSecond与timeout
                double permitsPerSecond = rateLimit.permitsPerSecond();
                long timeout = rateLimit.timeout();
                TimeUnit timeUnit = rateLimit.timeUnit();
                RateLimiter rateLimiter = null;
                if (!limitMap.containsKey(url)) {
                    rateLimiter = RateLimiter.create(permitsPerSecond);
                    limitMap.put(url, rateLimiter);
                    log.warn("请求======>" + url + "创建令牌桶,容量为:" + permitsPerSecond);
                }
                rateLimiter = limitMap.get(url);
                if (rateLimiter.tryAcquire(timeout, timeUnit)) {
                    obj = joinPoint.proceed();
                } else {
                    // 否则直接返回错误信息
                    obj = Objects.toString("请求太过频繁,请稍后重试");
                    log.warn("请求太过频繁,请稍后重试");
                }
            }
        } catch (Throwable e) {
            e.printStackTrace();
        }
        return obj;
    }

}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值