限流

1.限流的原理,根據(jù)ip(也可以對總的訪問頻次做限制)和固定時間內(nèi)的訪問次數(shù),做限制

2.springboot實現(xiàn)限流:

? ? 1)新建一個自定義注解:AccessLimit 用于接口上(注解實現(xiàn)更簡單方便)

? ? ? ??@Inherited

????????@Documented

????????@Target({ElementType.FIELD, ElementType.TYPE, ElementType.METHOD})

????????@Retention(RetentionPolicy.RUNTIME)

????????public @interface AccessLimit {

???????????? /**

? ????????? * 指定second 時間內(nèi),API最多的請求次數(shù)

????????????*/

????????????int times() default 3;

????????????/**

???????????? * 指定時間second,redis數(shù)據(jù)過期時間

???????????? */

????????????int second() default 10;

????????}

? ? 2)新建一個攔截器,獲取自定義注解上的參數(shù),然后緩存到redis中做校驗

????????@Component

????????public class AccessLimitInterceptimplements HandlerInterceptor {

????????private static final Loggerlogger = LoggerFactory.getLogger(AccessLimitIntercept.class);

? ? ????????@Autowired

? ????????? private RedisUtilredisUtil;

????????????@Override

????????????public boolean preHandle(HttpServletRequest request, HttpServletResponse response,Object handler)throws Exception{

????????????????try{

????????????????????// handler 是否為 HandlerMethod 實例

????????????????????if(handlerinstanceof HandlerMethod){

????????????????????????// 強轉(zhuǎn)

????????????????????????HandlerMethod handlerMethod = (HandlerMethod) handler;

????????????????????????// 獲取方法

????????????????????????Method method = handlerMethod.getMethod();

????????????????????????// 判斷方法是否有AccessLimit注解,有的才需要做限流

????????????????????????if(!method.isAnnotationPresent(AccessLimit.class)){

????????????????????????????return true;

????????????????????????}

????????????????????????// 獲取注釋上的內(nèi)容

????????????????????????AccessLimit accessLimit = method.getAnnotation(AccessLimit.class);

????????????????????????????if(accessLimit ==null){

????????????????????????????return true;

????????????????????????}

????????????????????????// 獲取方法上的請求次數(shù)

????????????????????????int times = accessLimit.times();

????????????????????????// 獲取方法注釋上的請求時間

????????????????????????Integer second = accessLimit.second();

????????????????????????// 拼接redis key = ip + api限流

????????????????????????String key = IpUtils.getIpAddr(request) + request.getRequestURI();

????????????????????????Integer maxTimes =null;

????????????????????????// 獲取 redis 的 value

????????????????????????String value =redisUtil.get(key) ==null?"":redisUtil.get(key).toString();

????????????????????????if(!"".equals(value)){

????????????????????????????maxTimes = Integer.valueOf(value);

????????????????????????}

????????????????????????if(maxTimes ==null){

????????????????????????????// 如果redis中沒有該ip對應(yīng)的時間則表示第一次調(diào)用,保存key到redis

????????????????????????????redisUtil.set(key,"1",second, TimeUnit.SECONDS);

????????????????????????}else if (maxTimes < times){

????????????????????????????// 如果 redis 中的時間比注解上的時間小,則表示可以允許訪問,這是修改redis的value時間

????????????????????????????redisUtil.set(key,maxTimes +1 +"",second,TimeUnit.SECONDS);

????????????????????????}else {

????????????????????????????logger.info(key +"請求過于頻繁");

????????????????????????????return setResponse(Result.error(ResultStatus.REQUEST_SEND_FREQUENTLY_ERROR.getStatus(),"請求過于頻繁,請稍后重試"),response);

????????????????????????}

????????????????????}

????????????????}catch (Exception ex){

????????????????????logger.error("API請求限流攔截異常,異常原因:" + ex);

????????????????????throw new ParameterException(ex.getMessage());

????????????????}

????????????????return true;

????????????}

????????????@Override

????????????public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView)throws Exception{

????????????}

????????????@Override

????????????public void afterCompletion(HttpServletRequest request,HttpServletResponse response,Object handler,Exception ex)throws Exception{

????????????}

????????????private boolean setResponse(Result result,HttpServletResponse response)throws IOException{

????????????????ServletOutputStream outputStream =null;

????????????????try{

????????????????????response.setHeader("Content-Type","application/json;charset=utf-8");

????????????????????outputStream = response.getOutputStream();

????????????????????outputStream.write(JSONUtil.toJsonStr(result).getBytes("UTF-8"));

????????????????}catch (Exception ex){

????????????????????logger.error("setResponse方法報錯",ex);

????????????????????return false;

? ? ? ? ? ? ? ? }finally {

????????????????????if(outputStream !=null){

????????????????????????outputStream.flush();

????????????????????????outputStream.close();

????????????????????}

? ? ? ? ? ? ? ? }

????????????????return false;

????????????}

????????}

? ? 3)配置攔截器

? ??????@Configuration

????????public class WebFilterConfig implements WebMvcConfigurer {

????????????/**

????????????* 這里需要先將限流攔截器入住,不然無法獲取到攔截器中的redistemplate

????????????* @return

????????????*/

????????????@Bean

????????????public AccessLimitIntercept getAccessLimitIntercept() {

????????????????return new AccessLimitIntercept();

????????????}

????????????/**

????????????* 多個攔截器組成一個攔截器鏈

????????????* @param registry

????????????*/

????????????@Override

????????????public void addInterceptors(InterceptorRegistry registry) {

????????????????registry.addInterceptor(getAccessLimitIntercept()).addPathPatterns("/**");

????????????}

????????}

? ? 4)在接口上添加注解,設(shè)置固定秒數(shù)內(nèi),限制的訪問次數(shù)

? ??????@RestController

????????@RequestMapping("/")

????????public class PingController extends BaseController {

????????????@AccessLimit(times = 5, second = 10)

????????????@GetMapping(value = "/ping")

????????????public Results ping() {

????????????????return succeed("pong", "");

????????????}

? ? ? ? }

? ? 5)注意:該限流加完后,接口依然可以被訪問,如果需要修改為接口不可被訪問,需要修改2)中setResponse()方法的放回值為false,preHandle()方法的返回值,可以決定方法是否可被執(zhí)行,當為false時,方法不會被執(zhí)行

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

友情鏈接更多精彩內(nèi)容