日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關咨詢
選擇下列產(chǎn)品馬上在線溝通
服務時間:8:30-17:00
你可能遇到了下面的問題
關閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
玩轉(zhuǎn)SpringMVC自定義請求匹配規(guī)則

環(huán)境:SpringBoot2.7.12

公司主營業(yè)務:成都網(wǎng)站設計、成都做網(wǎng)站、移動網(wǎng)站開發(fā)等業(yè)務。幫助企業(yè)客戶真正實現(xiàn)互聯(lián)網(wǎng)宣傳,提高企業(yè)的競爭能力。創(chuàng)新互聯(lián)是一支青春激揚、勤奮敬業(yè)、活力青春激揚、勤奮敬業(yè)、活力澎湃、和諧高效的團隊。公司秉承以“開放、自由、嚴謹、自律”為核心的企業(yè)文化,感謝他們對我們的高要求,感謝他們從不同領域給我們帶來的挑戰(zhàn),讓我們激情的團隊有機會用頭腦與智慧不斷的給客戶帶來驚喜。創(chuàng)新互聯(lián)推出都安免費做網(wǎng)站回饋大家。

前言

在Spring MVC框架中,HandlerMapping是用于將HTTP請求映射到處理器的方法的組件。當一個請求到達時,HandlerMapping會根據(jù)請求的URL和其他屬性來確定哪個處理器方法應該處理該請求。在Spring MVC中,我們可以自定義HandlerMapping來滿足特定的匹配需求。其中一個方法是使用getCustomMethodCondition()方法來自定義匹配條件。

本文將詳細介紹如何使用getCustomMethodCondition()方法來自定義HandlerMapping的匹配條件。通過閱讀本文,您將了解如何擴展HandlerMapping的默認行為,并使用自定義條件來匹配請求和處理器方法。

需求:我們希望根據(jù)請求header中的x-token值來匹配具體的接口。所有的接口都必須使用了自定義的注解標注。

1. 自定義請求匹配

在SpringMVC中可以通過自定義RequestMappingHandlerMapping#getCustomMethodCondition來實現(xiàn)此功能。

自定義請求匹配通過實現(xiàn)RequestCondition接口自定義規(guī)則

系統(tǒng)默認提供了以下RequestCondition實現(xiàn)

圖片

2. 自定義匹配條件

public class CustomRequestCondition implements RequestCondition {


  private static final String X_TOKEN_NAME = "x-token" ;


  private Method method ;


  public CustomRequestCondition(Method method) {
    this.method = method ;
  }


  // 當接口上有多個匹配規(guī)則時,進行合并操作
  @Override
  public CustomRequestCondition combine(CustomRequestCondition other) {
    return new CustomRequestCondition(other.method) ;
  }


  // 核心方法:根據(jù)匹配的條件進行判斷是否匹配,如果匹配則返回當前的對象,不匹配則返回null
  @Override
  public CustomRequestCondition getMatchingCondition(HttpServletRequest request) {
    AKF akf = method.getAnnotation(AKF.class) ;
    return akf != null ? buildToken(request, akf) : null ;
  }


  // 當有多個都滿足條件的時候,進行比較具體使用哪個
  @Override
  public int compareTo(CustomRequestCondition other, HttpServletRequest request) {
    return 0 ;
  }


  // 判斷請求header中的信息與注解中配置的信息是否一致
  private CustomRequestCondition buildToken(HttpServletRequest request, AKF akf) {
    String xToken = request.getHeader(X_TOKEN_NAME) ;
    if (xToken == null || xToken.length() == 0) {
      return null ;
    }
    return xToken.equals(akf.value()) ? this : null ;
  }


}

3. 配置自定義HandlerMapping

public class CustomMethodConditionRequestHandlerMapping extends RequestMappingHandlerMapping {
  @Override
  protected RequestCondition getCustomMethodCondition(Method method) {
    return new CustomRequestCondition(method) ;
  }
}

配置自定義的HandlerMapping

@Component
public class CustomEndpointConfig implements WebMvcRegistrations {
  public RequestMappingHandlerMapping getRequestMappingHandlerMapping() {
    return new CustomMethodConditionRequestHandlerMapping() ;
  }
}

通過實現(xiàn)WebMvcRegistrations中的getRequestMappingHandlerMapping方法覆蓋系統(tǒng)默認的RequestMappingHandlerMapping配置實現(xiàn)。當然這種方式你可能失去了某些功能。這里我們可以參考默認實現(xiàn)來完善自定義的實現(xiàn)。

4. 測試接口

@RestController
@RequestMapping("/conditions")
public class CustomMethodConditionController {


  @GetMapping("/index")
  public Object index() {
    return "custom method condition success" ;
  }


  @GetMapping("/index")
  @AKF
  public Object x() {
    return "x method invoke" ;
  }


  @GetMapping("/index")
  @AKF("x1")
  public Object x1() {
    return "x1 method invoke" ;
  }


  @GetMapping("/index")
  @AKF("x2")
  public Object x2() {
    return "x2 method invoke" ;
  }
}

上面的接口與通常的開發(fā)配置是一致的,只是有些有接口使用了@AKF注解。這些接口中,沒有@AKF注解或者沒有設置@AKF值的,都不能訪問,只有設置值了,且請求中攜帶了x-token并匹配上值了才會訪問到接口。

當訪問其它沒有@AKF注解的接口,返回404。

5. 原理

根據(jù)請求查找HandlerMethod

public abstract class AbstractHandlerMethodMapping extends AbstractHandlerMapping {
  protected HandlerMethod getHandlerInternal(HttpServletRequest request) throws Exception {
    String lookupPath = initLookupPath(request);
    try {
      // 根據(jù)請求查找匹配d餓HandlerMethod
      HandlerMethod handlerMethod = lookupHandlerMethod(lookupPath, request);
      return (handlerMethod != null ? handlerMethod.createWithResolvedBean() : null);
    }
  }
  protected HandlerMethod lookupHandlerMethod(String lookupPath, HttpServletRequest request) throws Exception {
    List matches = new ArrayList<>();
    // 根據(jù)請求的uri,獲取相應的RequestMappingInfo(該對象對應的Controller中的每一個接口)
    List directPathMatches = this.mappingRegistry.getMappingsByDirectPath(lookupPath);
    if (directPathMatches != null) {
      // 根據(jù)請求找到了相應的RequestMappingInfo,則進行匹配執(zhí)行相應的條件
      addMatchingMappings(directPathMatches, matches, request);
    }
    // ...
  }
  private void addMatchingMappings(Collection mappings, List matches, HttpServletRequest request) {
    for (T mapping : mappings) {
      // 執(zhí)行相應的條件進行匹配,比如:你在@RequestMapping中配置了header,params等相應的值
      T match = getMatchingMapping(mapping, request);
      if (match != null) {
        matches.add(new Match(match, this.mappingRegistry.getRegistrations().get(mapping)));
      }
    }
  }
}
public abstract class RequestMappingInfoHandlerMapping extends AbstractHandlerMethodMapping {
  protected RequestMappingInfo getMatchingMapping(RequestMappingInfo info, HttpServletRequest request) {
    return info.getMatchingCondition(request);
  }
}
// RequestMappingInfo
public final class RequestMappingInfo {
  // 該方法中就會根據(jù)請求request對象,判斷是否當前對象符合條件
  public RequestMappingInfo getMatchingCondition(HttpServletRequest request) {
    RequestMethodsRequestCondition methods = this.methodsCondition.getMatchingCondition(request);
    if (methods == null) {
      return null;
    }
    ParamsRequestCondition params = this.paramsCondition.getMatchingCondition(request);
    if (params == null) {
      return null;
    }
    HeadersRequestCondition headers = this.headersCondition.getMatchingCondition(request);
    if (headers == null) {
      return null;
    }


    // ...
    // 我們配置了自定義的,這里就會執(zhí)行我們自定義的條件(必須有@AKF注解)
    RequestConditionHolder custom = this.customConditionHolder.getMatchingCondition(request);
    if (custom == null) {
      // 返回null 則表示當前的RequestMappingInfo沒有匹配。
      // 最終如果都是返回的null,則最終返回客戶端將是404
      return null;
    }
    return new RequestMappingInfo(this.name, pathPatterns, patterns,
        methods, params, headers, consumes, produces, custom, this.options);
  }
}

在本文中,介紹了如何自定義RequestMappingHandlerMapping。通過自定義getCustomMethodCondition()方法,我們可以根據(jù)特定的需求擴展HandlerMapping的行為,并使用自定義條件來匹配請求和處理器方法。通過這種方式,我們可以更好地控制請求的處理邏輯。


網(wǎng)站名稱:玩轉(zhuǎn)SpringMVC自定義請求匹配規(guī)則
URL標題:http://www.dlmjj.cn/article/ccdjppo.html