成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

玩轉(zhuǎn)Spring MVC自定義請(qǐng)求匹配規(guī)則

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

環(huán)境:SpringBoot2.7.12

前言

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

本文將詳細(xì)介紹如何使用getCustomMethodCondition()方法來(lái)自定義HandlerMapping的匹配條件。通過(guò)閱讀本文,您將了解如何擴(kuò)展HandlerMapping的默認(rèn)行為,并使用自定義條件來(lái)匹配請(qǐng)求和處理器方法。

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

1. 自定義請(qǐng)求匹配

在SpringMVC中可以通過(guò)自定義RequestMappingHandlerMapping#getCustomMethodCondition來(lái)實(shí)現(xiàn)此功能。

自定義請(qǐng)求匹配通過(guò)實(shí)現(xiàn)RequestCondition接口自定義規(guī)則

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

圖片圖片

2. 自定義匹配條件

public class CustomRequestCondition implements RequestCondition<CustomRequestCondition> {


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


  private Method method ;


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


  // 當(dāng)接口上有多個(gè)匹配規(guī)則時(shí),進(jìn)行合并操作
  @Override
  public CustomRequestCondition combine(CustomRequestCondition other) {
    return new CustomRequestCondition(other.method) ;
  }


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


  // 當(dāng)有多個(gè)都滿足條件的時(shí)候,進(jìn)行比較具體使用哪個(gè)
  @Override
  public int compareTo(CustomRequestCondition other, HttpServletRequest request) {
    return 0 ;
  }


  // 判斷請(qǐng)求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() ;
  }
}

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

4. 測(cè)試接口

@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注解。這些接口中,沒(méi)有@AKF注解或者沒(méi)有設(shè)置@AKF值的,都不能訪問(wèn),只有設(shè)置值了,且請(qǐng)求中攜帶了x-token并匹配上值了才會(huì)訪問(wèn)到接口。

圖片

當(dāng)訪問(wèn)其它沒(méi)有@AKF注解的接口,返回404。

5. 原理

根據(jù)請(qǐng)求查找HandlerMethod

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


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

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

責(zé)任編輯:武曉燕 來(lái)源: Spring全家桶實(shí)戰(zhàn)案例源碼
相關(guān)推薦

2021-08-13 08:36:15

SpringMVC自定義

2016-08-23 13:21:15

MVC路由視圖

2023-12-28 08:22:33

響應(yīng)數(shù)據(jù)轉(zhuǎn)換

2022-11-10 07:53:54

Spring參數(shù)校驗(yàn)

2023-03-26 08:41:37

2022-11-01 11:15:56

接口策略模式

2024-10-14 17:18:27

2009-11-24 15:11:21

ASP.NET MVC

2009-07-22 15:27:39

ASP.NET MVC自定義路由

2011-03-17 09:45:01

Spring

2015-02-12 15:33:43

微信SDK

2010-04-30 09:32:49

ASP.NET MVC

2020-11-25 11:20:44

Spring注解Java

2016-11-16 21:55:55

源碼分析自定義view androi

2015-02-12 15:38:26

微信SDK

2011-06-23 10:49:13

Qt 自定義信號(hào)

2017-08-03 17:00:54

Springmvc任務(wù)執(zhí)行器

2022-06-20 08:26:39

Spring容器類型轉(zhuǎn)換

2016-12-26 15:25:59

Android自定義View

2021-05-12 08:32:53

Spring Secu 自定義session
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 国产精品视频一区二区三区, | 一区二区三区在线免费观看视频 | 特级黄一级播放 | 久草电影网| 国产在线一区观看 | 成人精品鲁一区一区二区 | 亚洲欧洲一区二区 | 一级片免费视频 | 久久久久久一区 | 韩三级在线观看 | 成人高清网站 | 黄色在线免费观看视频网站 | 日本超碰 | 狠狠视频 | 国产成视频在线观看 | 欧美free性| 日日干夜夜操 | 欧美成人精品激情在线观看 | 精品国产精品国产偷麻豆 | 色综合色综合色综合 | 极品粉嫩国产48尤物在线播放 | 欧美日韩国产一区二区 | 天天色影视综合 | 99精品视频免费观看 | 精品国产一区二区三区久久 | 日本不卡免费新一二三区 | 久久伊人精品 | 欧美aⅴ | 91精品国产91久久久久久最新 | 国产一区亚洲 | 日韩中文字幕在线视频 | 91亚洲国产成人久久精品网站 | 久久新| 午夜影院在线观看免费 | 综合色播 | 中文字幕不卡在线观看 | 天堂资源最新在线 | 天天躁日日躁狠狠躁2018小说 | 国产一区二区三区色淫影院 | 综合久久av | 国产日韩欧美电影 |