日日操夜夜添-日日操影院-日日草夜夜操-日日干干-精品一区二区三区波多野结衣-精品一区二区三区高清免费不卡

公告:魔扣目錄網(wǎng)為廣大站長(zhǎng)提供免費(fèi)收錄網(wǎng)站服務(wù),提交前請(qǐng)做好本站友鏈:【 網(wǎng)站目錄:http://www.ylptlb.cn 】, 免友鏈快審服務(wù)(50元/站),

點(diǎn)擊這里在線咨詢(xún)客服
新站提交
  • 網(wǎng)站:51998
  • 待審:31
  • 小程序:12
  • 文章:1030137
  • 會(huì)員:747

攔截器+redis

為了防止惡意訪問(wèn)接口造成服務(wù)器和數(shù)據(jù)庫(kù)壓力增大導(dǎo)致癱瘓,接口防刷(防止重復(fù)提交)在工作中是必不可少的,web項(xiàng)目前端也能夠?qū)崿F(xiàn),我們要介紹的是后端如何實(shí)現(xiàn)接口防刷。

實(shí)現(xiàn)思路

由于本人能力有限,只接觸過(guò)集群部署,一般都是使用兩種方案解決,一種是攔截器+Redis實(shí)現(xiàn),另外一種是使用攔截器+Guava Cache等本地緩存實(shí)現(xiàn),此處介紹第一種。

實(shí)現(xiàn)原理是利用攔截器攔截所有接口請(qǐng)求,然后對(duì)需要防刷的接口使用注解標(biāo)識(shí),在攔截器中判斷使用注解的方法,將根據(jù)請(qǐng)求的URI和用戶(hù)信息生成唯一的Key和訪問(wèn)次數(shù)存放到redis中,之后的每次請(qǐng)求都會(huì)使訪問(wèn)次數(shù)加一。

利用redis能夠過(guò)期的特性設(shè)定好一個(gè)訪問(wèn)周期的間隔時(shí)間。

實(shí)現(xiàn)目標(biāo):兩次請(qǐng)求時(shí)間間隔5秒不算重復(fù)提交,但30秒內(nèi)調(diào)用5次以上判定為惡意訪問(wèn)。

接下來(lái)我們來(lái)實(shí)現(xiàn)吧

具體實(shí)現(xiàn)

自定義一個(gè)注解AccessLimit,seconds為設(shè)置的秒數(shù)范圍,maxCount是范圍時(shí)間內(nèi)可以訪問(wèn)的次數(shù),needLogin與本文無(wú)關(guān)可忽略。

@Retention(RUNTIME)
@Target(METHOD)
public @interface AccessLimit {
    int seconds();
    int maxCount();
    boolean needLogin() default true;
}

創(chuàng)建一個(gè)攔截器,繼承HandlerInterceptorAdapter,在preHandle方法中做具體的操作。

每次請(qǐng)求都會(huì)根據(jù)key查詢(xún)r(jià)edis獲取其訪問(wèn)次數(shù),如果沒(méi)有則是第一次訪問(wèn),往redis中插入數(shù)據(jù),過(guò)期時(shí)間是注解中的屬性值seconds。

@Component
public class RepeatRequestInterceptor extends HandlerInterceptorAdapter {

    @Autowired
    private RedisUtils redisUtils;

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

        //判斷請(qǐng)求是否屬于方法的請(qǐng)求
        if(handler instanceof HandlerMethod){

            HandlerMethod hm = (HandlerMethod) handler;

            //獲取方法中的注解,看是否有該注解
            AccessLimit accessLimit = hm.getMethodAnnotation(AccessLimit.class);
            if(accessLimit == null){
                return true;
            }
            int seconds = accessLimit.seconds();
            int maxCount = accessLimit.maxCount();
            boolean login = accessLimit.needLogin();
            String key = request.getRequestURI();
            //如果需要登錄
            if(login){
                //獲取登錄的session進(jìn)行判斷
                //.....
                key+=""+"1";  //這里假設(shè)用戶(hù)是1,項(xiàng)目中是動(dòng)態(tài)獲取的userId
            }

            //從redis中獲取用戶(hù)訪問(wèn)的次數(shù)(redis中保存的key保存30秒,redisUtils使用的單位是秒)
            Integer count = redisUtils.get(key,Integer.class,seconds);
            if(count == null){
                //第一次訪問(wèn) key保存5秒 5秒后再訪問(wèn)key已過(guò)期,會(huì)重新生成
                redisUtils.set(key,1,5);
            }else if(count < maxCount){
                //加1
                redisUtils.increment(key);
            }else{
                //超出訪問(wèn)次數(shù)
                render(response);
                return false;
            }
        }
        return true;
    }
    private void render(HttpServletResponse response)throws Exception {
        response.setContentType("Application/json;charset=UTF-8");
        OutputStream out = response.getOutputStream();
        String str  = "{'mdg':'請(qǐng)求次數(shù)太多了'}";
        out.write(str.getBytes("UTF-8"));
        out.flush();
        out.close();
    }
}

附上redisUtils代碼

@Component
public class RedisUtils {
    @Autowired
    private RedisTemplate<String, Object> redisTemplate;
    @Autowired
    private ValueOperations<String, String> valueOperations;
    /**
     * 不設(shè)置過(guò)期時(shí)長(zhǎng)
     */
    public final static long NOT_EXPIRE = -1;

    /**
     * 設(shè)置key value
     */
    public void set(String key, Object value){
        set(key, value, CacheConstant.DEFAULT_EXPIRE);
    }

    public void set(String key, Object value, long expire){
        valueOperations.set(key, toJson(value));
        if(expire != NOT_EXPIRE){
            redisTemplate.expire(key, expire, TimeUnit.SECONDS);
        }
    }

    /**
     * 根據(jù)key獲得對(duì)象
     */
    public <T> T get(String key, Class<T> clazz) {
        return get(key, clazz, NOT_EXPIRE);
    }

    public <T> T get(String key, Class<T> clazz, long expire) {
        String value = valueOperations.get(key);
        if(expire != NOT_EXPIRE){
            redisTemplate.expire(key, expire, TimeUnit.SECONDS);
        }
        return value == null ? null : fromJson(value, clazz);
    }

    /**
     * 根據(jù)key獲得value
     */
    public String get(String key) {
        return get(key, NOT_EXPIRE);
    }
    public String get(String key, long expire) {
        String value = valueOperations.get(key);
        if(expire != NOT_EXPIRE){
            redisTemplate.expire(key, expire, TimeUnit.SECONDS);
        }
        return value;
    }

    public void increment(String key) {
        redisTemplate.opsForValue().increment(key,1L);
    }
}

通過(guò)JAVAconfig形式把Interceptor注冊(cè)到容器中

@Configuration
public class WebMvcConfig implements WebMvcConfigurer {
    @Autowired
    private RepeatRequestInterceptor interceptor;

   
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(interceptor);
    }

    @Override
    public void addCorsMappings(CorsRegistry registry) {
        registry.addMapping("/**")
                .allowedOrigins("*")
                .allowedMethods("GET", "HEAD", "POST","PUT", "DELETE", "OPTIONS")
                .allowCredentials(true).maxAge(3600);
    }
}

接口調(diào)用

編寫(xiě)一個(gè)測(cè)試類(lèi),寫(xiě)一個(gè)測(cè)試接口,如下

@RestController
@RequestMapping("/bid-applicant")
public class BidApplicantController extends BaseController {

    @AccessLimit(seconds=30, maxCount=5, needLogin=true)
    @RequestMapping("/fangshua")
    public ResponseInfo fangshua(){
        return ResponseInfo.ok("請(qǐng)求成功");
    }

測(cè)試

我在第一次請(qǐng)求后的30秒內(nèi)連續(xù)訪問(wèn)超過(guò)5次請(qǐng)求,會(huì)輸出我的報(bào)錯(cuò)信息,工作中可以跳轉(zhuǎn)自己的錯(cuò)誤頁(yè)面。

一招教你搞定API接口防刷

 


一招教你搞定API接口防刷

 

分享到:
標(biāo)簽:接口 API
用戶(hù)無(wú)頭像

網(wǎng)友整理

注冊(cè)時(shí)間:

網(wǎng)站:5 個(gè)   小程序:0 個(gè)  文章:12 篇

  • 51998

    網(wǎng)站

  • 12

    小程序

  • 1030137

    文章

  • 747

    會(huì)員

趕快注冊(cè)賬號(hào),推廣您的網(wǎng)站吧!
最新入駐小程序

數(shù)獨(dú)大挑戰(zhàn)2018-06-03

數(shù)獨(dú)一種數(shù)學(xué)游戲,玩家需要根據(jù)9

答題星2018-06-03

您可以通過(guò)答題星輕松地創(chuàng)建試卷

全階人生考試2018-06-03

各種考試題,題庫(kù),初中,高中,大學(xué)四六

運(yùn)動(dòng)步數(shù)有氧達(dá)人2018-06-03

記錄運(yùn)動(dòng)步數(shù),積累氧氣值。還可偷

每日養(yǎng)生app2018-06-03

每日養(yǎng)生,天天健康

體育訓(xùn)練成績(jī)?cè)u(píng)定2018-06-03

通用課目體育訓(xùn)練成績(jī)?cè)u(píng)定