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

詳解SpringMVC底層原理

原創 精選
開發 前端
筆者的專業是軟件技術,主修Java,記得剛開始寫Web應用的時候,都是直接寫Servlet,有多少個請求服務就寫多少個Servlet,于是一個系統中出現了一堆的Servlet,記得那會JSP也很流行,后來又經歷了Struts1、Struts2,到現在前后端技術分離了,則更多是用SpringMVC。

作者 | 波哥

審校 | 重樓

筆者的專業是軟件技術,主修Java,記得剛開始寫Web應用的時候,都是直接寫Servlet,有多少個請求服務就寫多少個Servlet,于是一個系統中出現了一堆的Servlet,記得那會JSP也很流行,后來又經歷了Struts1、Struts2,到現在前后端技術分離了,則更多是用SpringMVC。

隨著技術的發展,你會發現寫代碼變得越來越簡單,當然這個簡單是建立在前輩大神們深邃的設計思想上的,今天我們就來詳細聊聊SpringMVC,學習SpringMVC底層原理的同時,感受下大神們的設計思想。

我們先來通過一張圖了解下SpringMVC處理請求的整體流程:

使用過SpringMVC的老鐵們都知道,在SpringMVC中最核心的就是DispatcherServlet類,接下來筆者將從DispatcherServlet如何處理請求的整體流程來闡述它的底層實現。

DispatcherServlet毫無疑問是一個HttpServlet,因此可以追蹤到所有的請求都會進入到doDispatch方法中,而這個方法就是咱們要解剖的方法:

該方法主要有幾條主線:獲取HandlerExecutionChain、獲取HandlerAdapter、調用Adapter的handle方法、視圖處理器處理。下面我們將從這幾個主線逐一分析。

一、獲取HandlerExecutionChain

首先說明下這個HandlerExecutionChain,它里面封裝了一堆的Interceptor攔截器,以及Handler,它是一個處理鏈,通過getHandler方法獲取得到:

從代碼中我看到,這里循環handlerMappings,調用HandlerMapping的getHandler方法來獲取HandlerExecutionChain對象,獲取到了就返回,那這個handlerMappings都包含了哪些HandlerMapping呢?它們是什么時候被塞到handlerMappings集合中去的?

在DispatcherServlet的initStrategies方法中,會初始化一堆的數據,其中就有調用initHandlerMappings方法來初始化HandlerMapping,放到handlerMappings集合中,至于initStrategies方法是怎么被調用的,大家看下DispatcherServlet的繼承結構圖,然后根據Servlet的生命周期跟蹤下相信就知道了。

我們來看下initHandlerMappings方法:

從方法中可以看出,從Spring的Bean工廠中獲取HandlerMapping.class類型的Bean,以得到handlerMappings,在實際過程中,和咱們實際應用相關的HandlerMapping主要包括:BeanNameUrlHandlerMapping、SimpleUrlHandlerMapping、
RequestMappingHandlerMapping這三類,當然還有其他。

所以我們就知道了,上面是調用這三個HandlerMapping的getHandler方法來獲取HandlerExecutionChain對象。我們大概猜想,這個getHandler方法是根據請求的URL路徑,來獲取到處理的對象或者方法,這個是順理成章的,因為我們在開發的時候,就是通過配置path路徑來明確請求的路徑和方法或類的對應關系的。

我們先來看它們內部到底存放了什么,以及getHandler方法的具體邏輯,我們主要看BeanNameUrlHandlerMapping和
RequestMappingHandlerMapping。

1.BeanNameUrlHandlerMapping

從它的繼承關系圖中可以看到,它是Aware的實現類,跟蹤它的生命周期,Spring會調用initApplicationContext方法,然后調用detectHandlers方法來找到對應的Handler,并調用registerHandler方法將找到的Handler添加到handlerMap這個Map集合中去。

那上面"對應的Handler"需要滿足什么條件呢?

看上面的判斷邏輯,表明如果bean的名稱是以"/"開頭,則滿足條件,然后從Spring的Bean工廠中獲取到對應的Bean實例,添加到handlerMap集合。

在具體使用時,可以實現Controller和HttpRequestHandler接口,同時Component注解的value值以"/"開頭。

好,數據準備完成,接下來就是getHandler方法:

跟蹤代碼可以看出,就是根據URL從handlerMap獲取到對應的實例,隨后再將handler和HandlerInterceptor封裝成HandlerExecutionChain對象:

2.RequestMappingHandlerMapping

RequestMappingHandlerMapping是InitializingBean的實現類,在bean的初始化階段,它的afterPropertiesSet方法會被Spring調用,跟蹤該方法:

發現獲取所有的bean實例,然后循環調用processCandidateBean方法:

必須滿足一定條件的實例才會被處理,這個條件就是類上面包含Controller或者RequestMapping注解:

對于滿足上述條件的bean,會在detectHandlerMethods方法中將RequestMapping注解中的路徑和對應的方法封裝成HandlerMethod,并添加到mappingLookup集合中。

然后調用getHandler方法,根據請求URL從mappingLookup集合中取出HandlerMethod,并封裝成HandlerExecutionChain對象。

二、獲取HandlerAdapter

在上述調用getHandler方法,獲取到HandlerExecutionChain對象后,接下來調用getHandlerAdapter方法獲取HandlerAdapter:

循環所有的HandlerAdapter,調用supports方法判斷HandlerAdapter是否支持處理Handler。這里有兩個問題:第一、HandlerAdapter有哪些?它們是什么時候被初始化的?第二、每個HandlerAdapter的supports方法的具體實現;

1.HandlerAdapter有哪些?它們是什么時候被添加到handlerAdapters的?

同上,HandlerAdapter的初始化也是在initStrategies方法中發起的,在initHandlerAdapters方法中完成具體的添加:

可以看到,從Spring容器中獲取所有的HandlerAdapter類型的Bean添加到handlerAdapters中,默認情況下包括:
RequestMappingHandlerAdapter、HandlerFunctionAdapter、HttpRequestHandlerAdapter、SimpleControllerHandlerAdapter四個實現類的Bean實例。

2.HandlerAdapter的supports方法的具體實現

我們這里主要講下
RequestMappingHandlerAdapter、SimpleControllerHandlerAdapter和HttpRequestHandlerAdapter的supports實現邏輯:

  • RequestMappingHandlerAdapter

Handler是HandlerMethod類型,則由RequestMappingHandlerAdapter處理。

  • SimpleControllerHandlerAdapter

如果handler是Controller的實現類,則會由SimpleControllerHandlerAdapter處理。

  • HttpRequestHandlerAdapter

如果handler是HttpRequestHandler的實現類,則會由HttpRequestHandlerAdapter處理。

三、調用HandlerAdapter的handle方法

在獲取到HandlerAdapter后,還會調用interceptor的preHandle方法,這里就不詳細描述了。這里咱們直接看HandlerAdapter的handle方法的具體實現。

1.RequestMappingHandlerAdapter

相對于其他Adapter的處理方法,RequestMappingHandlerAdapter要復雜的的多

顯示創建ServletInvocableHandlerMethod對象,然后往對象中添加HandlerMethodArgumentResolvers和HandlerMethodReturnValueHandlers,這兩個接口很重要,是SpringMVC的兩個重要擴展點。

隨后開始處理,總結來說主要做了兩件事:

獲取方法參數值,然后調用方法:

執行方法相對簡單,咱們主要來看看是如何獲取方法參數值的:

先獲取到方法上的所有參數信息MethodParameter,然后調用
resolvers.supportsParameter方法來判斷是否支持對參數類型進行轉換,那這個resolvers是啥?它是一個HandlerMethodArgumentResolver,里面包含了一堆的HandlerMethodArgumentResolver,而這個HandlerMethodArgumentResolver就是專門負責參數轉換用的:

除了SpringMVC自己提供的HandlerMethodArgumentResolver外,還支持讓咱們自己來提供,只要實現HandlerMethodArgumentResolver即可。SpringMVC通過調用HandlerMethodArgumentResolver類的supportsParameter方法來找到一個適合處理的HandlerMethodArgumentResolver,找到了合適的Resolver后,調用它的resolveArgument方法來進行參數轉換,最終得到所有的參數值。

對返回值進行處理

在調用方法完成后,如果有返回值,則調用returnValueHandlers.handleReturnValue來處理返回值,這個returnValueHandlers是HandlerMethodReturnValueHandler類型的實例,它包含了一堆的HandlerMethodReturnValueHandler,HandlerMethodReturnValueHandler就是專門處理返回值的實現類,除了默認的HandlerMethodReturnValueHandler外,SpringMVC還允許咱們自己實現HandlerMethodReturnValueHandler。

首先SpringMVC會獲取一個最合適的HandlerMethodReturnValueHandler:

選擇的邏輯就是循環調用所有HandlerMethodReturnValueHandler的supportsReturnType方法,返回為true的就是最合適的:

得到HandlerMethodReturnValueHandler后,調用它的handleReturnValue方法來完成返回值的處理。

2.SimpleControllerHandlerAdapter


SimpleControllerHandlerAdapter的handle方法,就是執行Controller實現類的handleRequest方法。

3.HttpRequestHandlerAdapter

HttpRequestHandlerAdapter的handle方法,就是執行HttpRequestHandler實現類的handleRequest方法。

四、視圖渲染

上述完成Adapter的handle方法后,會執行過濾器HandlerInterceptor的postHandle方法,這里不再描述。如果返回值是ModelAndView,則會調用processDispatchResult,來完成視圖渲染:

這里會先得到一個View,也就是視圖器,然后調用view的render方法來完成渲染處理。那核心點就是如何獲取這個View。

循環調用viewResolvers中ViewResolver的resolveViewName方法,得到最合適的View。默認情況下,SpringMVC提供了四種類型的View:BeanNameViewResolver、ViewResolverComposite、
InternalResourceViewResolver、ContentNegotiatingViewResolver,當然咱們也可以自己實現ViewResolver,從而添加自己的View。

以上就是SpringMVC底層的大致實現原理,希望能給讀者朋友們一些幫助!

作者介紹

波哥,互聯行業從業10余年,先后擔任項目總監及架構師。目前專攻技術,喜歡研究技術原理。技術全面,主攻Java,精通JVM底層機制及Spring全家桶底層框架原理,熟練掌握當前主流的中間件、服務網格等技術原理。


責任編輯:華軒 來源: 51CTO
相關推薦

2022-12-19 08:00:00

SpringBootWeb開發

2023-07-11 08:00:00

2023-07-17 08:02:44

ZuulIO反應式

2021-02-20 08:40:19

HiveExplain底層

2023-07-20 10:04:37

底層路由配置

2021-08-29 07:41:48

數據HashMap底層

2021-08-31 07:36:22

LinkedListAndroid數據結構

2019-12-16 10:01:54

Java開發Web

2020-11-05 11:14:29

Docker底層原理

2023-01-04 07:54:03

HashMap底層JDK

2024-03-07 07:47:04

代碼塊Monitor

2023-10-18 10:55:55

HashMap

2021-07-05 07:51:43

JVM底層Python

2021-07-23 13:34:50

MySQL存儲InnoDB

2021-01-08 08:34:09

Synchronize線程開發技術

2023-02-12 23:23:30

2020-08-10 18:03:54

Cache存儲器CPU

2022-11-15 08:10:23

SpringMyBatis底層

2022-11-04 09:43:05

Java線程

2022-12-14 15:34:33

架構開發雙線程
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产精品无 | 午夜一级做a爰片久久毛片 精品综合 | 国产精品久久久久一区二区三区 | 日韩精品一区二区三区四区视频 | 91视频导航| 成人免费看片 | 日韩精品一区在线观看 | 狠狠色综合网站久久久久久久 | 一区二区三区高清 | 91精品国产91久久久久久吃药 | 免费国产一区二区 | 一区二区三区精品视频 | 2020天天操 | 久久精品91久久久久久再现 | 自拍偷拍亚洲一区 | 亚洲性网 | 成人国产精品久久 | 久久久精| 亚洲精品一区二区在线观看 | 久久久www成人免费精品 | 国产精品美女一区二区三区 | 亚洲二区视频 | 一区二区三区国产好 | 国产婷婷在线视频 | 亚洲一区二区中文字幕 | 在线观看中文字幕 | 成人午夜免费视频 | 日韩精品免费视频 | 日日综合 | 91视频a| 午夜精品久久久久久久久久久久 | 超碰超碰 | 最新毛片网站 | 国产欧美视频一区二区 | 国产在线视频一区二区董小宛性色 | 成人在线一区二区 | 性高湖久久久久久久久aaaaa | 亚洲91| 久久6视频| 国产精品久久久久久久久久免费 | 爱操影视|