欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring Cloud Zuul自定義過濾器的實現(xiàn)

 更新時間:2021年03月17日 08:35:48   作者:_燈火闌珊處  
這篇文章主要介紹了自定義Spring Cloud Zuul過濾器的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

構(gòu)建Zuul自定義過濾器,限制ip頻繁請求

自定義zuul過濾器其實很簡單

1. 首先pom文件得先引入zuul依賴

<dependency>
  <groupId>org.springframework.cloud</groupId>
  <artifactId>spring-cloud-starter-netflix-zuul</artifactId>
</dependency>

2. 創(chuàng)建一個類,繼承自ZuulFilter

import com.netflix.zuul.ZuulFilter;
import com.netflix.zuul.exception.ZuulException;
import org.springframework.stereotype.Component;

/**
 * 構(gòu)建zuul自定義過濾器
 */
@Component
public class MyFilter extends ZuulFilter {

  /**
   * 定義過濾器的類型
   * pre:在請求被路由之前執(zhí)行
   * route:在路由請求的時候執(zhí)行
   * post:請求路由以后執(zhí)行
   * error:處理請求時發(fā)生錯誤的時候執(zhí)行
   *
   * @return 過濾器的類型
   */
  @Override
  public String filterType() {
    return "pre";
  }

  /**
   * 過濾器執(zhí)行的順序,配置多個有順序的過濾
   * 執(zhí)行順序從小到大
   *
   * @return 執(zhí)行順序
   */
  @Override
  public int filterOrder() {
    return 1;
  }

  /**
   * 是否開啟過濾器
   * true:開啟
   * false:禁用
   *
   * @return 是否開啟過濾器
   */
  @Override
  public boolean shouldFilter() {
    return true;
  }

  /**
   * 過濾器的業(yè)務(wù)實現(xiàn)
   *
   * @return null 沒有意義
   * @throws ZuulException 異常信息
   */
  @Override
  public Object run() throws ZuulException {

    System.out.println("per zuul filter...");

    return null;
  }
}

自定義類上需要加上 @Component 注解
a. filterType()方法,定義過濾器的類型,返回的就是字符串,有以下4種類型

  • pre:在請求被路由之前執(zhí)行
  • route:在路由請求的時候執(zhí)行
  • post:請求路由以后執(zhí)行
  • error:處理請求時發(fā)生錯誤的時候執(zhí)行

b. filterOrder()方法,過濾器執(zhí)行的順序
c. shouldFilter()方法,是否開啟過濾器,true開啟,false不開啟
d. run()方法,過濾器的業(yè)務(wù)實現(xiàn),在這里寫實現(xiàn)邏輯的具體代碼

3. 限制ip頻繁請求,示例代碼

import com.imooc.grace.result.GraceJsonResult;
import com.imooc.grace.result.ResponseStatusEnum;
import com.imooc.utils.IPUtil;
import com.imooc.utils.JsonUtils;
import com.imooc.utils.RedisOperator;
import com.netflix.zuul.ZuulFilter;
import com.netflix.zuul.context.RequestContext;
import com.netflix.zuul.exception.ZuulException;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.MediaType;
import org.springframework.stereotype.Component;

import javax.servlet.http.HttpServletRequest;

/**
 * 限制ip頻繁請求
 */
@Component
public class BlackIpFilter extends ZuulFilter {

  /**
   * ip連續(xù)請求的次數(shù)
   */
  private static final int CONTINUE_COUNTS = 10;
  /**
   * ip判斷的時間間隔,單位秒
   */
  private static final int TIME_INTERVAL = 10;
  /**
   * 限制的時間,單位秒
   */
  private static final int LIMIT_TIMES = 15;

  @Autowired
  private RedisOperator redisOperator;

  @Override
  public String filterType() {
    return "pre";
  }


  @Override
  public int filterOrder() {
    // 這里設(shè)置為2,上面那個過濾器設(shè)置為1,則執(zhí)行順序為 1->2,大家可以測試一下
    return 2;
  }

  @Override
  public boolean shouldFilter() {
    return true;
  }

  @Override
  public Object run() throws ZuulException {
    // 獲取上下文對象
    RequestContext currentContext = RequestContext.getCurrentContext();
    HttpServletRequest request = currentContext.getRequest();
    // 獲取ip
    String requestIp = IPUtil.getRequestIp(request);

    // 判斷該ip在10秒內(nèi)請求次數(shù)是否超過10次,超過則限制該ip15秒內(nèi)不能訪問,15秒后再放行
    final String ipRedisKey = "zuul-ip:" + requestIp;
    final String ipRedisLimitKey = "zuul-ip-limit:" + requestIp;

    // 獲取當(dāng)前ip這個key的剩余時間
    long limitLeftTime = redisOperator.ttl(ipRedisLimitKey);
    // 判斷該ip是否還有剩余時間
    if (limitLeftTime > 0) {
      stopRequest(currentContext);
      return null;
    }

    // 在redis中累加ip的請求次數(shù)
    long requestCounts = redisOperator.increment(ipRedisKey, 1);
    if (requestCounts == 1) {
      redisOperator.expire(ipRedisKey, TIME_INTERVAL);
    }

    if (requestCounts > CONTINUE_COUNTS) {
      // 限制ip訪問
      redisOperator.set(ipRedisLimitKey, ipRedisLimitKey, LIMIT_TIMES);
      stopRequest(currentContext);
    }

    return null;
  }

  private void stopRequest(RequestContext context) {
    // 停止zuul繼續(xù)向下路由,禁止請求通信
    context.setSendZuulResponse(false);
    // 返回響應(yīng)碼200
    context.setResponseStatusCode(200);

    // TODO 要返回提示的json內(nèi)容(可以自定義任何響應(yīng)內(nèi)容)
    // 例如 {"status":544,"msg":"請求過于頻繁,請稍后再試","success":false,"data":null}
    String result = "json內(nèi)容";

    // 設(shè)置返回內(nèi)容
    context.setResponseBody(result);
    // 設(shè)置編碼
    context.getResponse().setCharacterEncoding("utf-8");
    // 設(shè)置返回內(nèi)容格式為json
    context.getResponse().setContentType(MediaType.APPLICATION_JSON_VALUE);
  }
}

這里使用了redis來記錄ip請求次數(shù)和控制時間間隔

獲取ip工具類 IPUtil

import javax.servlet.http.HttpServletRequest;

/**
 * 獲取ip工具類
 */
public class IPUtil {

  /**
   * 獲取請求IP:
   * 用戶的真實IP不能使用request.getRemoteAddr()
   * 這是因為可能會使用一些代理軟件,這樣ip獲取就不準確了
   * 此外我們?nèi)绻褂昧硕嗉墸↙VS/Nginx)反向代理的話,ip需要從X-Forwarded-For中獲得第一個非unknown的IP才是用戶的有效ip。
   */
  public static String getRequestIp(HttpServletRequest request) {
    String ip = request.getHeader("x-forwarded-for");
    if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
      ip = request.getHeader("Proxy-Client-IP");
    }
    if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
      ip = request.getHeader("WL-Proxy-Client-IP");
    }
    if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
      ip = request.getHeader("HTTP_CLIENT_IP");
    }
    if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
      ip = request.getHeader("HTTP_X_FORWARDED_FOR");
    }
    if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
      ip = request.getRemoteAddr();
    }
    return ip;
  }
}

到此這篇關(guān)于Spring Cloud Zuul自定義過濾器的實現(xiàn)的文章就介紹到這了,更多相關(guān)Spring Cloud Zuul過濾器內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 關(guān)于Java利用反射實現(xiàn)動態(tài)運行一行或多行代碼

    關(guān)于Java利用反射實現(xiàn)動態(tài)運行一行或多行代碼

    這篇文章主要介紹了關(guān)于Java利用反射實現(xiàn)動態(tài)運行一行或多行代碼,借鑒了別人的方法和書上的內(nèi)容,最后將題目完成了,和大家一起分享以下解決方法,需要的朋友可以參考下
    2023-04-04
  • Java容器ArrayList知識點總結(jié)

    Java容器ArrayList知識點總結(jié)

    本篇文章給大家分享了Java容器ArrayList的相關(guān)知識點,對此有需要的朋友可以跟著學(xué)習(xí)參考下。
    2018-05-05
  • Java實現(xiàn)FTP上傳到服務(wù)器

    Java實現(xiàn)FTP上傳到服務(wù)器

    這篇文章主要為大家詳細介紹了Java實現(xiàn)FTP上傳到服務(wù)器,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-09-09
  • springboot?集成identityserver4身份驗證的過程解析

    springboot?集成identityserver4身份驗證的過程解析

    這篇文章主要介紹了springboot?集成identityserver4身份驗證的相關(guān)知識,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友參考下吧
    2024-01-01
  • Javabean轉(zhuǎn)換成json字符并首字母大寫代碼實例

    Javabean轉(zhuǎn)換成json字符并首字母大寫代碼實例

    這篇文章主要介紹了javabean轉(zhuǎn)成json字符并首字母大寫代碼實例,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-02-02
  • Java深入解析接口interface

    Java深入解析接口interface

    接口是Java中最重要的概念之一,它可以被理解為一種特殊的類,不同的是接口的成員沒有執(zhí)行體,是由全局常量和公共的抽象方法所組成,本文給大家介紹Java接口,感興趣的朋友一起看看吧
    2022-06-06
  • Springboot內(nèi)置tomcat配置虛擬路徑過程解析

    Springboot內(nèi)置tomcat配置虛擬路徑過程解析

    這篇文章主要介紹了Springboot內(nèi)置tomcat配置虛擬路徑過程解析,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-04-04
  • java打印指定年月的日歷

    java打印指定年月的日歷

    這篇文章主要為大家詳細介紹了Java如何打印指定年月的日歷,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-07-07
  • Spring+Http請求+HttpClient實現(xiàn)傳參

    Spring+Http請求+HttpClient實現(xiàn)傳參

    這篇文章主要介紹了Spring+Http請求+HttpClient實現(xiàn)傳參,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-03-03
  • 淺談一下Java中的內(nèi)存模型JMM

    淺談一下Java中的內(nèi)存模型JMM

    這篇文章主要介紹了淺談一下Java中的內(nèi)存模型JMM,JMM,全程是?Java?Memory?Model?,直譯就是?Java?內(nèi)存模型,根據(jù)這個名字,可以知道它是?Java?設(shè)計用來管理內(nèi)存的一個模型,需要的朋友可以參考下
    2023-08-08

最新評論