亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

HandlerMapping之RequestMappingHandlerMapping作用詳解

 更新時(shí)間:2023年10月12日 09:10:58   作者:Evan_L  
這篇文章主要介紹了HandlerMapping之RequestMappingHandlerMapping作用詳解,HandlerMapping是用來尋找Handler的,并不與Handler的類型或者實(shí)現(xiàn)綁定,而是根據(jù)需要定義的,那么為什么要單獨(dú)給@RequestMapping實(shí)現(xiàn)一個(gè)HandlerMapping,需要的朋友可以參考下

前言

上回我們知道HandlerMapping是用來尋找Handler的,并不與Handler的類型或者實(shí)現(xiàn)綁定,而是根據(jù)需要定義的。

那么為什么要單獨(dú)給@RequestMapping實(shí)現(xiàn)一個(gè)HandlerMapping?這次咱們就來專門看看這個(gè)RequestMappingHandlerMapping。

RequestMappingHandlerMapping

名字來源

因?yàn)镽equestMappingHandlerMapping是專門為@RequestMapping而生的,因此他的名字是這樣來的:@RequestMapping的HandlerMapping了。

為什么不叫MethodHandlerMapping呢?

主要還是Handler是一個(gè)邏輯概念,MethodHandler了只是對(duì)目標(biāo)方法進(jìn)行了封裝,并不是真正處理請(qǐng)求的。

真正處理請(qǐng)求的是我們@RequestMapping的方法。取個(gè)名字都給你講道理。

@RequestMapping

在解答文章開頭的問題前,我們先看看@RequestMapping

@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Mapping
public @interface RequestMapping {
	/**
	 * 處理器的名字,支持類級(jí)別和方法級(jí)別,多個(gè)路徑用#分割
	 */
	String name() default "";
	/**
	 * 匹配請(qǐng)求路徑
	 */
	@AliasFor("path")
	String[] value() default {};
	/**
	 * 匹配請(qǐng)求路徑
	 */
	@AliasFor("value")
	String[] path() default {};
	/**
	 * Http請(qǐng)求方法:可選GET, POST, HEAD, OPTIONS, PUT, PATCH, DELETE, TRACE
	 */
	RequestMethod[] method() default {};
	/**
	 * 匹配指定的地址欄參數(shù)
	 */
	String[] params() default {};
	/**
	 * 匹配特定的header
	 */
	String[] headers() default {};
	/**
	 * 匹配特定的Content-Type
	 */
	String[] consumes() default {};
	/**
	 * 匹配特定的Accept
	 */
	String[] produces() default {};
}

發(fā)現(xiàn)了嗎,各位?他除了能根據(jù)URI匹配,還能根據(jù)請(qǐng)求頭、請(qǐng)求方法、甚至是請(qǐng)求參數(shù)來匹配!

上回說的基于URL的兩種HandlerMapping都不能滿足他,因此必須推出一個(gè)更加強(qiáng)大、可擴(kuò)展性更強(qiáng)的HandlerMapping——RequestMappingHandlerMapping

在這里插入圖片描述

那么問題來了:

  • @RequestMapping是在什么如何被解析的呢?
    • 我們很容易想到的就是,遍歷容器中所有的對(duì)象,檢查是否存在@Controller注解。存在,那就是控制器。然后接著遍歷所有聲明的public方法,檢查是否存在@RequestMapping方法。這樣,我們就找到了處理器方法。
  • @RequestMapping是在什么時(shí)候被解析的呢?
    • 本著“誰使用,誰解析”的原則,他自然是被RequestMappingHandlerMapping解析的。而又因?yàn)锧RequestMapping的尋找可太費(fèi)功夫,不可能在提供映射服務(wù)時(shí)再來解析,只能是初始化時(shí)進(jìn)行解析。因此實(shí)現(xiàn)InitializingBean進(jìn)行初始化,是個(gè)選擇。

沒錯(cuò),實(shí)際上,SpringMVC跟你想的一樣。在InitializingBean的afterPropertiesSet方法中,完成了以3下件事:

  1. 尋找@Controller的bean,并找到所有的@RequestMapping方法
  2. 解析@RequestMapping封裝成RequestMappingInfo
  3. 將以上解析到的信息進(jìn)行注冊(cè)。

信息包括:

信息描述
@Controller/@RequestMapping對(duì)象反射調(diào)用目標(biāo)方法時(shí),需要的target對(duì)象
RequestMappingInfo由@RequestMapping解析而來
@RequestMapping的方法注冊(cè)時(shí),注冊(cè)器會(huì)將Method與handler對(duì)象一起封裝成HandlerMethod進(jìn)行注冊(cè)。便于后面適配器調(diào)用。

@RequestMapping的注冊(cè)

前面的解析@RequestMapping到RequestMappingInfo,可以省略,比較簡(jiǎn)單。但是@RequestMapping的注冊(cè)沒辦法省略。因?yàn)槿绻悴磺宄窃趺醋?cè)的,也就沒辦法理解他是怎么尋找目標(biāo)處理器的。

為了支撐@RequestMapping多樣化的匹配條件,不能再像前面兩款HanderMapping一樣,簡(jiǎn)單粗暴的使用Map了。在 org.springframework.web.servlet.handler.AbstractHandlerMethodMapping 的內(nèi)部定義了內(nèi)部類,專門用來做注冊(cè)中心,管理映射關(guān)系。

/**
	 * Mapping注冊(cè)中心,可以理解為辦事處
	 */
	class MappingRegistry {
		/**
		 * T是匹配條件的對(duì)象。MappingRegistration是注冊(cè)的信息,可以理解為你要做事情。
		 * 對(duì)于RequestMappingHandlerMapping,T就是RequestMappingInfo
		 * MappingRegistration包括信息:RequestMappingInfo、HandlerMethod、directPaths、mappingName、corsConfig
		 */
		private final Map<T, MappingRegistration<T>> registry = new HashMap<>();
		/**
		 * Map<path, RequestMappingInfo>
		 */
		private final MultiValueMap<String, T> pathLookup = new LinkedMultiValueMap<>();
		/**
		 * Map<mappingName, List<HandlerMethod>>
		 */
		private final Map<String, List<HandlerMethod>> nameLookup = new ConcurrentHashMap<>();
	}

從他的屬性,我們可以分析得到如下信息:

  1. 直接通過請(qǐng)求路徑來查找處理器時(shí),需要經(jīng)過pathLookup中轉(zhuǎn)registry,最后拿到MappingRegistration才到達(dá)HandlerMethod.
  2. 直接通過mappingName則可以直接找到HandlerMethod. 不過這個(gè)是Spring為了支持 <%@ taglib uri="//www.springframework.org/tags" prefix="s" %> 而提供的。跟我們平時(shí)使用沒多大關(guān)系。

但是各位觀眾老爺,pathLookup找到的是一個(gè),而mappingName能找到多個(gè),這是咋回事?誤會(huì)啊,pathLookup的value可不是簡(jiǎn)單的一個(gè)元素,而是多個(gè)!他是MultiValueMap,不是我們經(jīng)??吹降牡?cái)傌汬ashMap。他可以一個(gè)key對(duì)應(yīng)多個(gè)value。

但是為什么會(huì)有多個(gè)呢?或者說為什么需要保存多個(gè)呢? 因?yàn)橐粋€(gè)Handler可以處理多個(gè)請(qǐng)求,如果由多個(gè)Handler都能處理某一個(gè)請(qǐng)求的時(shí)候怎么辦呢?況且SpringMVC還支持通配符匹配。umm…這一幕有點(diǎn)似曾相識(shí),我們的nginx做路由轉(zhuǎn)發(fā)的時(shí)候,不是也有類似的問題嗎?這意味著在查找Handler的時(shí)候,我們還需要找到最佳的匹配。例如,/*相較于/hello,那肯定是/hello更精確,更合適啦。你看,多嚴(yán)謹(jǐn)!

總結(jié)

由于@RequestMapping支持靈活的請(qǐng)求匹配條件,而不只是簡(jiǎn)單的路徑,只能開發(fā)出RequestMappingHandlerMapping進(jìn)行支持。

RequestMappingHandlerMapping是通過InitializingBean進(jìn)行初始化的,在這里完成@RequestMappingHandlerMapping的掃描和解析,以及注冊(cè)。

HandlerMethod是在注冊(cè)時(shí)進(jìn)行封裝的。獲取Handler時(shí),拿到的Handler就是HandlerMethod。

后面的適配器適配的,也是他。

RequestMappingHandlerMapping使用了InitializingBean做初始化,但是當(dāng)我們自己在做初始化的時(shí)候,尤其是使用多種初始化方式的時(shí)候,應(yīng)當(dāng)要注意Spring的調(diào)用順序,否則有可能發(fā)生NPE,或者獲取不到目標(biāo)屬性的情況。

例如:同時(shí)在ApplicationContextAware、InitializingBean、@PostConstruct進(jìn)行初始化。 為此,給大家找了官方的bean的生命周期

在這里插入圖片描述

到此這篇關(guān)于HandlerMapping之RequestMappingHandlerMapping作用詳解的文章就介紹到這了,更多相關(guān)RequestMappingHandlerMapping作用內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Spring Security 安全框架應(yīng)用原理解析

    Spring Security 安全框架應(yīng)用原理解析

    這篇文章主要介紹了Spring Security 安全框架應(yīng)用,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-07-07
  • java多線程實(shí)現(xiàn)有序輸出ABC

    java多線程實(shí)現(xiàn)有序輸出ABC

    這篇文章主要為大家詳細(xì)介紹了java多線程實(shí)現(xiàn)有序輸出ABC,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-08-08
  • java之向linux文件夾下寫文件無權(quán)限的問題

    java之向linux文件夾下寫文件無權(quán)限的問題

    這篇文章主要介紹了java之向linux文件夾下寫文件無權(quán)限的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-09-09
  • Windows部署Jar包的三種方式圖文詳解

    Windows部署Jar包的三種方式圖文詳解

    使用Java編寫了一些有用的工具,因?yàn)椴环奖悴渴鸬椒?wù)器上,所以需要把Java生成的jar包在本地Windows上部署,這篇文章主要給大家介紹了關(guān)于Windows部署Jar包的三種方式,需要的朋友可以參考下
    2023-07-07
  • SpringBoot如何通過自定義注解實(shí)現(xiàn)權(quán)限檢查詳解

    SpringBoot如何通過自定義注解實(shí)現(xiàn)權(quán)限檢查詳解

    這篇文章主要給大家介紹了關(guān)于SpringBoot如何通過自定義注解實(shí)現(xiàn)權(quán)限檢查的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-10-10
  • SpringBoot三種方法接口返回日期格式化小結(jié)

    SpringBoot三種方法接口返回日期格式化小結(jié)

    本文介紹了三種在Spring Boot中格式化接口返回日期的方法,包含使用@JsonFormat注解、全局配置JsonConfig、以及在yml文件中配置時(shí)區(qū),具有一定的參考價(jià)值,感興趣的可以了解一下
    2025-01-01
  • 基于Java判斷網(wǎng)絡(luò)是否正常代碼實(shí)例

    基于Java判斷網(wǎng)絡(luò)是否正常代碼實(shí)例

    這篇文章主要介紹了基于Java判斷網(wǎng)絡(luò)是否正常代碼實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-03-03
  • springboot2?使用activiti6?idea插件的過程詳解

    springboot2?使用activiti6?idea插件的過程詳解

    這篇文章主要介紹了springboot2?使用activiti6?idea插件,本文通過截圖實(shí)例代碼相結(jié)合給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-03-03
  • Java利用泛型實(shí)現(xiàn)折半查找法

    Java利用泛型實(shí)現(xiàn)折半查找法

    泛型是JAVA重要的特性,使用泛型編程,可以使代碼復(fù)用率提高。查找作為泛型的一個(gè)簡(jiǎn)單應(yīng)用,本文將使用泛型實(shí)現(xiàn)折半查找法,感興趣的可以了解一下
    2022-08-08
  • springboot國(guó)際化多語言配置方式

    springboot國(guó)際化多語言配置方式

    這篇文章主要介紹了springboot國(guó)際化多語言配置方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-04-04

最新評(píng)論