日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 前端技术 > javascript >内容正文

javascript

SpringMVC深度探险(三) —— DispatcherServlet与初始化主线

發(fā)布時(shí)間:2025/5/22 javascript 26 豆豆
生活随笔 收集整理的這篇文章主要介紹了 SpringMVC深度探险(三) —— DispatcherServlet与初始化主线 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

本文是專欄文章(SpringMVC深度探險(xiǎn))系列的文章之一,博客地址為:http://downpour.iteye.com/blog/1341459

在上一篇文章中,我們給出了構(gòu)成SpringMVC應(yīng)用程序的三要素以及三要素的設(shè)計(jì)過(guò)程。讓我們來(lái)歸納一下整個(gè)設(shè)計(jì)過(guò)程中的一些要點(diǎn):

  • SpringMVC將Http處理流程抽象為一個(gè)又一個(gè)處理單元
  • SpringMVC定義了一系列組件(接口)與所有的處理單元對(duì)應(yīng)起來(lái)
  • SpringMVC由DispatcherServlet貫穿始終,并將所有的組件串聯(lián)起來(lái)

在整個(gè)過(guò)程中,組件和DispatcherServlet總是維持著一個(gè)相互支撐的關(guān)系:

  • DispatcherServlet —— 串聯(lián)起整個(gè)邏輯主線,是整個(gè)框架的心臟
  • 組件 —— 邏輯處理單元的程序化表示,起到承上啟下的作用,是SpringMVC行為模式的實(shí)際承載者

在本系列接下來(lái)的兩篇文章中,我們將分別討論DispatcherServlet和組件的相關(guān)內(nèi)容。本文討論DispatcherServlet,而下一篇?jiǎng)t重點(diǎn)分析組件。

有關(guān)DispatcherServlet,我們想從構(gòu)成DispatcherServlet的體系結(jié)構(gòu)入手,再根據(jù)不同的邏輯主線分別加以分析,希望能夠幫助讀者整理出學(xué)習(xí)SpringMVC核心類的思路。

DispatcherServlet的體系結(jié)構(gòu)

通過(guò)不同的角度來(lái)觀察DispatcherServlet會(huì)得到不同的結(jié)論。我們?cè)谶@里選取了三個(gè)不同的角度:運(yùn)行特性、繼承結(jié)構(gòu)和數(shù)據(jù)結(jié)構(gòu)。

【運(yùn)行主線】

從DispatcherServlet所實(shí)現(xiàn)的接口來(lái)看,DispatcherServlet的核心本質(zhì):是一個(gè)Servlet。這個(gè)結(jié)論似乎很幼稚,不過(guò)這個(gè)幼稚的結(jié)論卻蘊(yùn)含了一個(gè)對(duì)整個(gè)框架都至關(guān)重要的內(nèi)在原則:Servlet可以根據(jù)其特性進(jìn)行運(yùn)行主線的劃分

根據(jù)Servlet規(guī)范的定義,Servlet中的兩大核心方法init方法和service方法,它們的運(yùn)行時(shí)間和觸發(fā)條件都截然不同:

1. init方法

在整個(gè)系統(tǒng)啟動(dòng)時(shí)運(yùn)行,且只運(yùn)行一次。因此,在init方法中我們往往會(huì)對(duì)整個(gè)應(yīng)用程序進(jìn)行初始化操作。這些初始化操作可能包括對(duì)容器(WebApplicationContext)的初始化、組件和外部資源的初始化等等。

2. service方法

在整個(gè)系統(tǒng)運(yùn)行的過(guò)程中處于偵聽模式,偵聽并處理所有的Web請(qǐng)求。因此,在service及其相關(guān)方法中,我們看到的則是對(duì)Http請(qǐng)求的處理流程。

因而在這里,Servlet的這一特性就被SpringMVC用于對(duì)不同的邏輯職責(zé)加以劃分,從而形成兩條互不相關(guān)的邏輯運(yùn)行主線:

  • 初始化主線 —— 負(fù)責(zé)對(duì)SpringMVC的運(yùn)行要素進(jìn)行初始化
  • Http請(qǐng)求處理主線 —— 負(fù)責(zé)對(duì)SpringMVC中的組件進(jìn)行邏輯調(diào)度完成對(duì)Http請(qǐng)求的處理

對(duì)于一個(gè)MVC框架而言,運(yùn)行主線的劃分非常重要。因?yàn)橹挥信宄煌倪\(yùn)行主線,我們才能針對(duì)不同的運(yùn)行主線采取不同的研究策略。而我們?cè)谶@個(gè)系列中的絕大多數(shù)分析的切入點(diǎn),也是圍繞著不同的運(yùn)行主線進(jìn)行的。

:SpringMVC運(yùn)行主線的劃分依據(jù)是Servlet對(duì)象中不同方法的生命周期。事實(shí)上,幾乎所有的MVC都是以此為依據(jù)來(lái)進(jìn)行運(yùn)行主線的劃分。這進(jìn)一步可以證明所有的MVC框架的核心基礎(chǔ)還是Servlet規(guī)范,而設(shè)計(jì)理念的差異也導(dǎo)致了不同的框架走向了完全不同的發(fā)展道路。

【繼承結(jié)構(gòu)】

除了運(yùn)行主線的劃分以外,我們?cè)訇P(guān)注一下DispatcherServlet的繼承結(jié)構(gòu):



在這個(gè)繼承結(jié)構(gòu)中,我們可以看到DispatcherServlet在其繼承樹中包含了2個(gè)Spring的支持類:HttpServletBean和FrameworkServlet。我們分別來(lái)討論一下這兩個(gè)Spring的支持類在這里所起到的作用。

HttpServletBean是Spring對(duì)于Servlet最低層次的抽象。在這一層抽象中,Spring會(huì)將這個(gè)Servlet視作是一個(gè)Spring的bean,并將init-param中的值作為bean的屬性注入進(jìn)來(lái):

Java代碼 ?

  • public?final?void?init()?throws?ServletException?{??
  • ????if?(logger.isDebugEnabled())?{??
  • ????????logger.debug("Initializing?servlet?'"?+?getServletName()?+?"'");??
  • ????}??
  • ??
  • ????//?Set?bean?properties?from?init?parameters.??
  • ????try?{??
  • ????????PropertyValues?pvs?=?new?ServletConfigPropertyValues(getServletConfig(),?this.requiredProperties);??
  • ????????BeanWrapper?bw?=?PropertyAccessorFactory.forBeanPropertyAccess(this);??
  • ????????ResourceLoader?resourceLoader?=?new?ServletContextResourceLoader(getServletContext());??
  • ????????bw.registerCustomEditor(Resource.class,?new?ResourceEditor(resourceLoader,?this.environment));??
  • ????????initBeanWrapper(bw);??
  • ????????bw.setPropertyValues(pvs,?true);??
  • ????}??
  • ????catch?(BeansException?ex)?{??
  • ????????logger.error("Failed?to?set?bean?properties?on?servlet?'"?+?getServletName()?+?"'",?ex);??
  • ????????throw?ex;??
  • ????}??
  • ??
  • ????//?Let?subclasses?do?whatever?initialization?they?like.??
  • ????initServletBean();??
  • ??
  • ????if?(logger.isDebugEnabled())?{??
  • ????????logger.debug("Servlet?'"?+?getServletName()?+?"'?configured?successfully");??
  • ????}??
  • }??
  • public final void init() throws ServletException {

    ????if (logger.isDebugEnabled()) {

    ????????logger.debug("Initializing servlet '" + getServletName() + "'");

    ????}

    ?

    ????// Set bean properties from init parameters.

    ????try {

    ????????PropertyValues pvs = new ServletConfigPropertyValues(getServletConfig(), this.requiredProperties);

    ????????BeanWrapper bw = PropertyAccessorFactory.forBeanPropertyAccess(this);

    ????????ResourceLoader resourceLoader = new ServletContextResourceLoader(getServletContext());

    ????????bw.registerCustomEditor(Resource.class, new ResourceEditor(resourceLoader, this.environment));

    ????????initBeanWrapper(bw);

    ????????bw.setPropertyValues(pvs, true);

    ????}

    ????catch (BeansException ex) {

    ????????logger.error("Failed to set bean properties on servlet '" + getServletName() + "'", ex);

    ????????throw ex;

    ????}

    ?

    ????// Let subclasses do whatever initialization they like.

    ????initServletBean();

    ?

    ????if (logger.isDebugEnabled()) {

    ????????logger.debug("Servlet '" + getServletName() + "' configured successfully");

    ????}

    }



    從源碼中,我們可以看到HttpServletBean利用了Servlet的init方法的執(zhí)行特性,將一個(gè)普通的Servlet與Spring的容器聯(lián)系在了一起。在這其中起到核心作用的代碼是:BeanWrapper bw = PropertyAccessorFactory.forBeanPropertyAccess(this);將當(dāng)前的這個(gè)Servlet類轉(zhuǎn)化為一個(gè)BeanWrapper,從而能夠以Spring的方式來(lái)對(duì)init-param的值進(jìn)行注入。BeanWrapper的相關(guān)知識(shí)屬于Spring Framework的內(nèi)容,我們?cè)谶@里不做詳細(xì)展開,讀者可以具體參考HttpServletBean的注釋獲得更多的信息。

    FrameworkServlet則是在HttpServletBean的基礎(chǔ)之上的進(jìn)一步抽象。通過(guò)FrameworkServlet真正初始化了一個(gè)Spring的容器(WebApplicationContext),并引入到Servlet對(duì)象之中:

    Java代碼 ?

  • protected?final?void?initServletBean()?throws?ServletException?{??
  • ????getServletContext().log("Initializing?Spring?FrameworkServlet?'"?+?getServletName()?+?"'");??
  • ????if?(this.logger.isInfoEnabled())?{??
  • ????????this.logger.info("FrameworkServlet?'"?+?getServletName()?+?"':?initialization?started");??
  • ????}??
  • ????long?startTime?=?System.currentTimeMillis();??
  • ??
  • ????try?{??
  • ????????this.webApplicationContext?=?initWebApplicationContext();??
  • ????????initFrameworkServlet();??
  • ????}?catch?(ServletException?ex)?{??
  • ????????this.logger.error("Context?initialization?failed",?ex);??
  • ????????throw?ex;??
  • ????}?catch?(RuntimeException?ex)?{??
  • ????????this.logger.error("Context?initialization?failed",?ex);??
  • ????????throw?ex;??
  • ????}??
  • ??
  • ????if?(this.logger.isInfoEnabled())?{??
  • ????????long?elapsedTime?=?System.currentTimeMillis()?-?startTime;??
  • ????????this.logger.info("FrameworkServlet?'"?+?getServletName()?+?"':?initialization?completed?in?"?+??
  • ????????????????elapsedTime?+?"?ms");??
  • ????}??
  • }??
  • protected final void initServletBean() throws ServletException {

    ????getServletContext().log("Initializing Spring FrameworkServlet '" + getServletName() + "'");

    ????if (this.logger.isInfoEnabled()) {

    ????????this.logger.info("FrameworkServlet '" + getServletName() + "': initialization started");

    ????}

    ????long startTime = System.currentTimeMillis();

    ?

    ????try {

    ????????this.webApplicationContext = initWebApplicationContext();

    ????????initFrameworkServlet();

    ????} catch (ServletException ex) {

    ????????this.logger.error("Context initialization failed", ex);

    ????????throw ex;

    ????} catch (RuntimeException ex) {

    ????????this.logger.error("Context initialization failed", ex);

    ????????throw ex;

    ????}

    ?

    ????if (this.logger.isInfoEnabled()) {

    ????????long elapsedTime = System.currentTimeMillis() - startTime;

    ????????this.logger.info("FrameworkServlet '" + getServletName() + "': initialization completed in " +

    ????????????????elapsedTime + " ms");

    ????}

    }



    上面的這段代碼就是FrameworkServlet初始化的核心代碼。從中我們可以看到這個(gè)FrameworkServlet將調(diào)用其內(nèi)部的方法initWebApplicationContext()對(duì)Spring的容器(WebApplicationContext)進(jìn)行初始化。同時(shí),FrameworkServlet還暴露了與之通訊的結(jié)構(gòu)可供子類調(diào)用:

    Java代碼 ?

  • public?abstract?class?FrameworkServlet?extends?HttpServletBean?{??
  • ??
  • ????/**?WebApplicationContext?for?this?servlet?*/??
  • ????private?WebApplicationContext?webApplicationContext;??
  • ??
  • ????????//?這里省略了其他所有的代碼??
  • ??
  • ????/**?
  • ?????*?Return?this?servlet's?WebApplicationContext.?
  • ?????*/??
  • ????public?final?WebApplicationContext?getWebApplicationContext()?{??
  • ????????return?this.webApplicationContext;??
  • ????}??
  • }??
  • public abstract class FrameworkServlet extends HttpServletBean {

    ?

    ????/** WebApplicationContext for this servlet */

    ????private WebApplicationContext webApplicationContext;

    ?

    // 這里省略了其他所有的代碼

    ?

    ????/**

    ???? * Return this servlet's WebApplicationContext.

    ???? */

    ????public final WebApplicationContext getWebApplicationContext() {

    ????????return this.webApplicationContext;

    ????}

    }



    我們?cè)谶@里暫且不對(duì)Spring容器(WebApplicationContext)的初始化過(guò)程詳加探查,稍后我們會(huì)討論一些WebApplicationContext初始化過(guò)程中的配置選項(xiàng)。不過(guò)讀者可以在這里體會(huì)到:FrameworkServlet在其內(nèi)部初始化了一個(gè)Spring的容器(WebApplicationContext)并暴露了相關(guān)的操作接口,因而繼承自FrameworkServlet的DispatcherServlet,也就直接擁有了與WebApplicationContext進(jìn)行通信的能力。

    通過(guò)對(duì)DispatcherServlet繼承結(jié)構(gòu)的研究,我們可以明確:

    downpour 寫道

    結(jié)論 DispatcherServlet的繼承體系架起了DispatcherServlet與Spring容器進(jìn)行溝通的橋梁。



    【數(shù)據(jù)結(jié)構(gòu)】

    在上一篇文章中,我們?cè)?jīng)提到過(guò)DispatcherServlet的數(shù)據(jù)結(jié)構(gòu):



    我們可以把在上面這張圖中所構(gòu)成DispatcherServlet的數(shù)據(jù)結(jié)構(gòu)主要分為兩類(我們?cè)谶@里用一根分割線將其分割開來(lái)):

    • 配置參數(shù) —— 控制SpringMVC組件的初始化行為方式
    • 核心組件 —— SpringMVC的核心邏輯處理組件

    可以看到,這兩類數(shù)據(jù)結(jié)構(gòu)都與SpringMVC中的核心要素組件有關(guān)。因此,我們可以得出這樣一個(gè)結(jié)論:

    downpour 寫道

    結(jié)論 組件是整個(gè)DispatcherServlet的靈魂所在:它不僅是初始化主線中的初始化對(duì)象,同樣也是Http請(qǐng)求處理主線中的邏輯調(diào)度載體。



    :我們可以看到被我們劃為配置參數(shù)的那些變量都是boolean類型的,它們將在DispatcherServlet的初始化主線中起到一定的作用,我們?cè)谥髸?huì)使用源碼進(jìn)行說(shuō)明。而這些boolean值可以通過(guò)web.xml中的init-param值進(jìn)行設(shè)定覆蓋(這是由HttpServletBean的特性帶來(lái)的)。

    SpringMVC的運(yùn)行體系

    DispatcherServlet繼承結(jié)構(gòu)和數(shù)據(jù)結(jié)構(gòu),實(shí)際上表述的是DispatcherServlet與另外兩大要素之間的關(guān)系:

    • 繼承結(jié)構(gòu) —— DispatcherServlet與Spring容器(WebApplicationContext)之間的關(guān)系
    • 數(shù)據(jù)結(jié)構(gòu) —— DispatcherServlet與組件之間的關(guān)系

    所以,其實(shí)我們可以這么說(shuō):SpringMVC的整個(gè)運(yùn)行體系,是由DispatcherServlet、組件和容器這三者共同構(gòu)成的。

    在這個(gè)運(yùn)行體系中,DispatcherServlet是邏輯處理的調(diào)度中心,組件則是被調(diào)度的操作對(duì)象。而容器在這里所起到的作用,是協(xié)助DispatcherServlet更好地對(duì)組件進(jìn)行管理。這就相當(dāng)于一個(gè)工廠招了一大批的工人,并把工人劃分到一個(gè)統(tǒng)一的工作車間而便于管理。在工廠要進(jìn)行生產(chǎn)活動(dòng)時(shí),只需要從工作車間把工人分派到相應(yīng)的生產(chǎn)流水線上即可。

    筆者在這里引用Spring官方reference中的一幅圖,對(duì)三者之間的關(guān)系進(jìn)行簡(jiǎn)單的描述:



    :在這幅圖中,我們除了看到在圖的左半邊DispatcherServlet、組件和容器這三者之間的調(diào)用關(guān)系以外,還可以看到SpringMVC的運(yùn)行體系與其它運(yùn)行體系之間存在著關(guān)系。有關(guān)這一點(diǎn),我們?cè)谥蟮挠懻撝袝?huì)詳細(xì)展開。

    既然是三個(gè)元素之間的關(guān)系表述,我們必須以兩兩關(guān)系的形式進(jìn)行歸納:

    • DispatcherServlet - 容器 —— DispatcherServlet對(duì)容器進(jìn)行初始化
    • 容器 - 組件 —— 容器對(duì)組件進(jìn)行全局管理
    • DispatcherServlet - 組件 —— DispatcherServlet對(duì)組件進(jìn)行邏輯調(diào)用

    值得注意的是,在上面這幅圖中,三大元素之間的兩兩關(guān)系其實(shí)表現(xiàn)得并不明顯,尤其是"容器 - 組件"和"DispatcherServlet - 組件"之間的關(guān)系。這主要是由于Spring官方reference所給出的這幅圖是一個(gè)靜態(tài)的關(guān)系表述,如果從動(dòng)態(tài)的觀點(diǎn)來(lái)對(duì)整個(gè)過(guò)程加以審視,我們就不得不將SpringMVC的運(yùn)行體系與之前所提到的運(yùn)行主線聯(lián)系在一起,看看這些元素在不同的邏輯主線中所起到的作用。

    接下來(lái),我們就分別看看DispatcherServlet的兩條運(yùn)行主線。

    DispatcherServlet的初始化主線

    對(duì)于DispatcherServlet的初始化主線,我們首先應(yīng)該明確幾個(gè)基本觀點(diǎn):

    • 初始化主線的驅(qū)動(dòng)要素 —— servlet中的init方法
    • 初始化主線的執(zhí)行次序 —— HttpServletBean -> FrameworkServlet -> DispatcherServlet
    • 初始化主線的操作對(duì)象 —— Spring容器(WebApplicationContext)和組件

    這三個(gè)基本觀點(diǎn),可以說(shuō)是我們對(duì)之前所有討論的一個(gè)小結(jié)。明確了這些內(nèi)容,我們就可以更加深入地看看DispatcherServlet初始化主線的過(guò)程:



    在這幅圖中,我們站在一個(gè)動(dòng)態(tài)的角度將DispatcherServlet、容器(WebApplicationContext)和組件這三者之間的關(guān)系表述出來(lái),同時(shí)給出了這三者之間的運(yùn)行順序和邏輯過(guò)程。讀者或許對(duì)其中的絕大多數(shù)細(xì)節(jié)還很陌生,甚至有一種無(wú)從下手的感覺(jué)。這沒(méi)有關(guān)系,大家可以首先抓住圖中的執(zhí)行線,回憶一下之前有關(guān)DispatcherServlet的繼承結(jié)構(gòu)和數(shù)據(jù)結(jié)構(gòu)的內(nèi)容。接下來(lái),我們就圖中的內(nèi)容逐一進(jìn)行解釋。

    【W(wǎng)ebApplicationContext的初始化】

    之前我們討論了DispatcherServlet對(duì)于WebApplicationContext的初始化是在FrameworkServlet中完成的,不過(guò)我們并沒(méi)有細(xì)究其中的細(xì)節(jié)。在默認(rèn)情況下,這個(gè)初始化過(guò)程是由web.xml中的入口程序配置所驅(qū)動(dòng)的:

    Xml代碼 ?

  • <!--?Processes?application?requests?-->??
  • <servlet>??
  • ????<servlet-name>dispatcher</servlet-name>??
  • ????<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>??
  • ????<load-on-startup>1</load-on-startup>??
  • </servlet>??
  • ??????????
  • <servlet-mapping>??
  • ????<servlet-name>dispatcher</servlet-name>??
  • ????<url-pattern>/**</url-pattern>??
  • </servlet-mapping>??
  • <!-- Processes application requests -->

    <servlet>

    ????<servlet-name>dispatcher</servlet-name>

    ????<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>

    ????<load-on-startup>1</load-on-startup>

    </servlet>

    ????????

    <servlet-mapping>

    ????<servlet-name>dispatcher</servlet-name>

    ????<url-pattern>/**</url-pattern>

    </servlet-mapping>



    我們已經(jīng)不止一次提到過(guò)這段配置,不過(guò)在這之前都沒(méi)有對(duì)這段配置做過(guò)什么很詳細(xì)的分析。事實(shí)上,這段入口程序的配置中隱藏了SpringMVC的兩大要素(核心分發(fā)器Dispatcher和核心配置文件[servlet-name]-servlet.xml)之間的關(guān)系表述:

    downpour 寫道

    在默認(rèn)情況下,web.xml配置節(jié)點(diǎn)中<servlet-name>的值就是建立起核心分發(fā)器DispatcherServlet與核心配置文件之間聯(lián)系的橋梁。DispatcherServlet在初始化時(shí)會(huì)加載位置在/WEB-INF/[servlet-name]-servlet.xml的配置文件作為SpringMVC的核心配置。



    SpringMVC在這里采用了一個(gè)"命名約定"的方法進(jìn)行關(guān)系映射,這種方法很廉價(jià)也很管用。以上面的配置為例,我們就必須在/WEB-INF/目錄下,放一個(gè)名為dispatcher-servlet.xml的Spring配置文件作為SpringMVC的核心配置用以指定SpringMVC的基本組件聲明定義。

    這看上去似乎有一點(diǎn)別扭,因?yàn)樵趯?shí)際項(xiàng)目中,我們通常喜歡把配置文件放在classpath下,并使用不同的package進(jìn)行區(qū)分。例如,在基于Maven的項(xiàng)目結(jié)構(gòu)中,所有的配置文件應(yīng)置于src/main/resources目錄下,這樣才比較符合配置文件統(tǒng)一化管理的最佳實(shí)踐。

    于是,Spring提供了一個(gè)初始化的配置選項(xiàng),通過(guò)指定contextConfigLocation選項(xiàng)來(lái)自定義SpringMVC核心配置文件的位置:

    Xml代碼 ?

  • <!--?Processes?application?requests?-->??
  • <servlet>??
  • ????<servlet-name>dispatcher</servlet-name>??
  • ????<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>??
  • ????<init-param>??
  • ????????<param-name>contextConfigLocation</param-name>??
  • ????????<param-value>classpath:web/applicationContext-dispatcherServlet.xml</param-value>??
  • ????</init-param>??
  • ????<load-on-startup>1</load-on-startup>??
  • </servlet>??
  • ??????????
  • <servlet-mapping>??
  • ????<servlet-name>dispatcher</servlet-name>??
  • ????<url-pattern>/</url-pattern>??
  • </servlet-mapping>??
  • <!-- Processes application requests -->

    <servlet>

    ????<servlet-name>dispatcher</servlet-name>

    ????<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>

    ????<init-param>

    ????????<param-name>contextConfigLocation</param-name>

    ????????<param-value>classpath:web/applicationContext-dispatcherServlet.xml</param-value>

    ????</init-param>

    ????<load-on-startup>1</load-on-startup>

    </servlet>

    ????????

    <servlet-mapping>

    ????<servlet-name>dispatcher</servlet-name>

    ????<url-pattern>/</url-pattern>

    </servlet-mapping>



    這樣一來(lái),DispatcherServlet在初始化時(shí),就會(huì)自動(dòng)加載在classpath下,web這個(gè)package下名為applicationContext-dispatcherServlet.xml的文件作為其核心配置并用以初始化容器(WebApplicationContext)。

    當(dāng)然,這只是DispatcherServlet在進(jìn)行WebApplicationContext初始化過(guò)程中的配置選項(xiàng)之一。我們可以在Spring的官方reference中找到相應(yīng)的配置選項(xiàng),有興趣的讀者可以參照reference的說(shuō)明進(jìn)行嘗試:



    所有的這些配置選項(xiàng),實(shí)際上都是為了讓DispatcherServlet對(duì)WebApplicationContext的初始化過(guò)程顯得更加自然。不過(guò)這只是完成了容器(WebApplicationContext)的構(gòu)建工作,那么容器所管理的那些組件,又是如何進(jìn)行初始化的呢?

    downpour 寫道

    結(jié)論 SpringMVC核心配置文件中所有的bean定義,就是SpringMVC的組件定義,也是DispatcherServlet在初始化容器(WebApplicationContext)時(shí),所要進(jìn)行初始化的組件。



    在上一篇文章我們談到組件的時(shí)候就曾經(jīng)提到,SpringMVC自身對(duì)于組件并未實(shí)現(xiàn)一套完整的管理機(jī)制,而是借用了Spring Framework核心框架中容器的概念,將所有的組件納入到容器中進(jìn)行管理。所以,SpringMVC的核心配置文件使用與傳統(tǒng)Spring Framework相同的配置形式,而整個(gè)管理的體系也是一脈相承的。

    :Spring3.0之后,單獨(dú)為SpringMVC建立了專用的schema,從而使得我們可以使用Schema Based XML來(lái)對(duì)SpringMVC的組件進(jìn)行定義。不過(guò)我們可以將其視作是傳統(tǒng)Spring配置的一個(gè)補(bǔ)充,而不要過(guò)于糾結(jié)不同的配置格式。

    我們知道,SpringMVC的組件是一個(gè)個(gè)的接口定義,當(dāng)我們?cè)赟pringMVC的核心配置文件中定義一個(gè)組件時(shí),使用的卻是組件的實(shí)現(xiàn)類:

    Xml代碼 ?

  • <bean?class="org.springframework.web.servlet.view.InternalResourceViewResolver">??
  • ????<property?name="prefix"?value="/"?/>??
  • ????<property?name="suffix"?value=".jsp"?/>??
  • </bean>??
  • <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">

    ????<property name="prefix" value="/" />

    ????<property name="suffix" value=".jsp" />

    </bean>



    這也就是Spring管理組件的模式:用具體的實(shí)現(xiàn)類來(lái)指定接口的行為方式。不同的實(shí)現(xiàn)類,代表著不同的組件行為模式,它們?cè)赟pring容器中是可以共存的:

    Xml代碼 ?

  • <bean?class="org.springframework.web.servlet.view.InternalResourceViewResolver">??
  • ????<property?name="prefix"?value="/"?/>??
  • ????<property?name="suffix"?value=".jsp"?/>??
  • </bean>??
  • ??
  • <bean?class="org.springframework.web.servlet.view.freemarker.FreeMarkerViewResolver">??
  • ????<property?name="prefix"?value="/"?/>??
  • ????<property?name="suffix"?value=".ftl"?/>??
  • </bean>??
  • <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">

    ????<property name="prefix" value="/" />

    ????<property name="suffix" value=".jsp" />

    </bean>

    ?

    <bean class="org.springframework.web.servlet.view.freemarker.FreeMarkerViewResolver">

    ????<property name="prefix" value="/" />

    ????<property name="suffix" value=".ftl" />

    </bean>



    所以,Spring的容器就像是一個(gè)聚寶盆,它只負(fù)責(zé)承載對(duì)象,管理好對(duì)象的生命周期,而并不關(guān)心一個(gè)組件接口到底有多少種實(shí)現(xiàn)類或者行為模式。這也就是我們?cè)谏厦婺欠鶊D中,畫了多個(gè)HandlerMappings、HandlerAdapters和ViewResolvers的原因:一個(gè)組件的多種行為模式可以在容器中共存,容器將負(fù)責(zé)對(duì)這些實(shí)現(xiàn)類進(jìn)行管理。而具體如何使用這些對(duì)象,則由應(yīng)用程序自身來(lái)決定。

    如此一來(lái),我們可以大致概括一下WebApplicationContext初始化的兩個(gè)邏輯層次:

    • DispatcherServlet負(fù)責(zé)對(duì)容器(WebApplicationContext)進(jìn)行初始化。
    • 容器(WebApplicationContext)將讀取SpringMVC的核心配置文件進(jìn)行組件的實(shí)例化。

    整個(gè)過(guò)程,我們把應(yīng)用程序的日志級(jí)別調(diào)低,可以進(jìn)行非常詳細(xì)的觀察:

    引用

    14:15:27,037 DEBUG StandardServletEnvironment:100 - Initializing new StandardServletEnvironment
    14:15:27,128 DEBUG DispatcherServlet:115 - Initializing servlet 'dispatcher'
    14:15:27,438? INFO DispatcherServlet:444 - FrameworkServlet 'dispatcher': initialization started

    14:15:27,449 DEBUG DispatcherServlet:572 - Servlet with name 'dispatcher' will try to create custom WebApplicationContext context of class 'org.springframework.web.context.support.XmlWebApplicationContext', using parent context [null]
    1571 [main] INFO /sample - Initializing Spring FrameworkServlet 'dispatcher'
    14:15:27,505 DEBUG StandardServletEnvironment:100 - Initializing new StandardServletEnvironment
    14:15:27,546? INFO XmlWebApplicationContext:495 - Refreshing WebApplicationContext for namespace 'dispatcher-servlet': startup date [Mon Feb 06 14:15:27 CST 2012]; root of context hierarchy
    14:15:27,689? INFO XmlBeanDefinitionReader:315 - Loading XML bean definitions from class path resource [web/applicationContext-dispatcherServlet.xml]
    14:15:27,872 DEBUG PluggableSchemaResolver:140 - Loading schema mappings from [META-INF/spring.schemas]
    14:15:28,442 DEBUG PathMatchingResourcePatternResolver:550 - Looking for matching resources in directory tree [D:\Work\Demo2do\Sample\target\classes\com\demo2do\sample\web\controller]
    14:15:28,442 DEBUG PathMatchingResourcePatternResolver:612 - Searching directory [D:\Work\Demo2do\Sample\target\classes\com\demo2do\sample\web\controller] for files matching pattern [D:/Work/Demo2do/Sample/target/classes/com/demo2do/sample/web/controller/**/*.class]
    14:15:28,450 DEBUG PathMatchingResourcePatternResolver:351 - Resolved location pattern [classpath*:com/demo2do/sample/web/controller/**/*.class] to resources [file [D:\Work\Demo2do\Sample\target\classes\com\demo2do\sample\web\controller\BlogController.class], file [D:\Work\Demo2do\Sample\target\classes\com\demo2do\sample\web\controller\UserController.class]]
    14:15:28,569 DEBUG ClassPathBeanDefinitionScanner:243 - Identified candidate component class: file [D:\Work\Demo2do\Sample\target\classes\com\demo2do\sample\web\controller\BlogController.class]
    14:15:28,571 DEBUG ClassPathBeanDefinitionScanner:243 - Identified candidate component class: file [D:\Work\Demo2do\Sample\target\classes\com\demo2do\sample\web\controller\UserController.class]
    14:15:28,634 DEBUG BeanDefinitionParserDelegate:497 - Neither XML 'id' nor 'name' specified - using generated bean name [org.springframework.web.servlet.view.InternalResourceViewResolver#0]
    14:15:28,635 DEBUG XmlBeanDefinitionReader:216 - Loaded 19 bean definitions from location pattern [classpath:web/applicationContext-dispatcherServlet.xml]
    14:15:28,635 DEBUG XmlWebApplicationContext:525 - Bean factory for WebApplicationContext for namespace 'dispatcher-servlet': org.springframework.beans.factory.support.DefaultListableBeanFactory@2321b59a: defining beans [org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping#0,org.springframework.format.support.FormattingConversionServiceFactoryBean#0,org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter#0,org.springframework.web.servlet.handler.MappedInterceptor#0,org.springframework.web.servlet.mvc.method.annotation.ExceptionHandlerExceptionResolver#0,org.springframework.web.servlet.mvc.annotation.ResponseStatusExceptionResolver#0,org.springframework.web.servlet.mvc.support.DefaultHandlerExceptionResolver#0,org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping,org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter,org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter,blogController,userController,org.springframework.context.annotation.internalConfigurationAnnotationProcessor,org.springframework.context.annotation.internalAutowiredAnnotationProcessor,org.springframework.context.annotation.internalRequiredAnnotationProcessor,org.springframework.context.annotation.internalCommonAnnotationProcessor,org.springframework.web.servlet.resource.ResourceHttpRequestHandler#0,org.springframework.web.servlet.handler.SimpleUrlHandlerMapping#0,org.springframework.web.servlet.view.InternalResourceViewResolver#0]; root of factory hierarchy
    14:15:29,015 DEBUG RequestMappingHandlerMapping:98 - Looking for request mappings in application context: WebApplicationContext for namespace 'dispatcher-servlet': startup date [Mon Feb 06 14:15:27 CST 2012]; root of context hierarchy
    14:15:29,037? INFO RequestMappingHandlerMapping:188 - Mapped "{[/blog],methods=[],params=[],headers=[],consumes=[],produces=[],custom=[]}" onto public org.springframework.web.servlet.ModelAndView com.demo2do.station.web.controller.BlogController.index()
    14:15:29,039? INFO RequestMappingHandlerMapping:188 - Mapped "{[/login],methods=[],params=[],headers=[],consumes=[],produces=[],custom=[]}" onto public org.springframework.web.servlet.ModelAndView com.demo2do.station.web.controller.UserController.login(java.lang.String,java.lang.String)
    14:15:29,040 DEBUG DefaultListableBeanFactory:458 - Finished creating instance of bean 'org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping#0'
    14:15:29,460 DEBUG BeanNameUrlHandlerMapping:71 - Looking for URL mappings in application context: WebApplicationContext for namespace 'dispatcher-servlet': startup date [Mon Feb 06 14:15:27 CST 2012]; root of context hierarchy
    14:15:29,539 DEBUG DefaultListableBeanFactory:458 - Finished creating instance of bean 'org.springframework.web.servlet.resource.ResourceHttpRequestHandler#0'
    14:15:29,540 DEBUG DefaultListableBeanFactory:217 - Creating shared instance of singleton bean 'org.springframework.web.servlet.handler.SimpleUrlHandlerMapping#0'
    14:15:29,555? INFO SimpleUrlHandlerMapping:314 - Mapped URL path [/static/**] onto handler 'org.springframework.web.servlet.resource.ResourceHttpRequestHandler#0'
    14:15:29,556 DEBUG DefaultListableBeanFactory:458 - Finished creating instance of bean 'org.springframework.web.servlet.handler.SimpleUrlHandlerMapping#0'
    14:15:29,827 DEBUG DispatcherServlet:523 - Published WebApplicationContext of servlet 'dispatcher' as ServletContext attribute with name [org.springframework.web.servlet.FrameworkServlet.CONTEXT.dispatcher]
    14:15:29,827? INFO DispatcherServlet:463 - FrameworkServlet 'dispatcher': initialization completed in 2389 ms
    14:15:29,827 DEBUG DispatcherServlet:136 - Servlet 'dispatcher' configured successfully

    4047 [main] INFO org.mortbay.log - Started SelectChannelConnector@0.0.0.0:8080
    Jetty Server started, use 4267 ms



    在這段啟動(dòng)日志(筆者進(jìn)行了一定刪減)中,筆者刻意將WebApplicationContext的初始化的標(biāo)志日志使用紅色的標(biāo)進(jìn)行區(qū)分,而將核心配置文件的讀取位置和組件定義初始化的標(biāo)志日志使用藍(lán)色標(biāo)記加以區(qū)分。相信有了這段日志的幫助,讀者應(yīng)該可以對(duì)WebApplicationContext的初始化過(guò)程有了更加直觀的認(rèn)識(shí)。

    :啟動(dòng)日志是我們研究SpringMVC的主要途徑之一,之后我們還將反復(fù)使用這種方法對(duì)SpringMVC的運(yùn)行過(guò)程進(jìn)行研究。讀者應(yīng)該仔細(xì)品味每一條日志的作用,從而能夠與之后的分析講解呼應(yīng)起來(lái)。

    或許讀者對(duì)WebApplicationContext對(duì)組件進(jìn)行初始化的過(guò)程還有點(diǎn)困惑,大家不妨先將這個(gè)過(guò)程省略,把握住整個(gè)DispatcherServlet的大方向。我們?cè)谥蟮奈恼轮?#xff0c;還將對(duì)SpringMVC的組件、這些組件的定義以及組件的初始化方式做進(jìn)一步的分析和探討。

    到此為止,圖中順著FrameworkServlet的那些箭頭,我們已經(jīng)交代清楚,讀者可以回味一下整個(gè)過(guò)程。

    【獨(dú)立的WebApplicationContext體系】

    獨(dú)立的WebApplicationContext體系,是SpringMVC初始化主線中的一個(gè)非常重要的概念。回顧一下剛才曾經(jīng)提到過(guò)的DispatcherServlet、容器和組件三者之間的關(guān)系,我們?cè)谝玫哪歉惫俜絩eference的示意圖中,實(shí)際上已經(jīng)包含了這一層意思:

    downpour 寫道

    結(jié)論 在DispatcherServlet初始化的過(guò)程中所構(gòu)建的WebApplicationContext獨(dú)立于Spring自身的所構(gòu)建的其他WebApplicationContext體系而存在。



    稍有一些Spring編程經(jīng)驗(yàn)的程序員,對(duì)于下面的配置應(yīng)該非常熟悉:

    Xml代碼 ?

  • <context-param>??
  • ????<param-name>contextConfigLocation</param-name>??
  • ????<param-value>classpath:context/applicationContext-*.xml</param-value>??
  • </context-param>??
  • ??????
  • <listener>??
  • ????<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>??
  • </listener>??
  • <context-param>

    ????<param-name>contextConfigLocation</param-name>

    ????<param-value>classpath:context/applicationContext-*.xml</param-value>

    </context-param>

    ????

    <listener>

    ????<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>

    </listener>



    在上面的代碼中,我們定義了一個(gè)Listener,它會(huì)在整個(gè)Web應(yīng)用程序啟動(dòng)的時(shí)候運(yùn)行一次,并初始化傳統(tǒng)意義上的Spring的容器。這也是一般情況下,當(dāng)并不使用SpringMVC作為我們的表示層解決方案,卻希望在我們的Web應(yīng)用程序中使用Spring相關(guān)功能時(shí)所采取的一種配置方式。

    如果我們要在這里引入SpringMVC,整個(gè)配置看上去就像這樣:

    Xml代碼 ?

  • <context-param>??
  • ????<param-name>contextConfigLocation</param-name>??
  • ????<param-value>classpath:context/applicationContext-*.xml</param-value>??
  • </context-param>??
  • ??????
  • <listener>??
  • ????<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>??
  • </listener>??
  • ??????
  • <!--?Processes?application?requests?-->??
  • <servlet>??
  • ????<servlet-name>dispatcher</servlet-name>??
  • ????<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>??
  • ????<init-param>??
  • ????????<param-name>contextConfigLocation</param-name>??
  • ????????<param-value>classpath:web/applicationContext-dispatcherServlet.xml</param-value>??
  • ????</init-param>??
  • ????<load-on-startup>1</load-on-startup>??
  • </servlet>??
  • ??????????
  • <servlet-mapping>??
  • ????<servlet-name>dispatcher</servlet-name>??
  • ????<url-pattern>/</url-pattern>??
  • </servlet-mapping>??
  • <context-param>

    ????<param-name>contextConfigLocation</param-name>

    ????<param-value>classpath:context/applicationContext-*.xml</param-value>

    </context-param>

    ????

    <listener>

    ????<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>

    </listener>

    ?

    <!-- Processes application requests -->

    <servlet>

    ????<servlet-name>dispatcher</servlet-name>

    ????<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>

    ????<init-param>

    ????????<param-name>contextConfigLocation</param-name>

    ????????<param-value>classpath:web/applicationContext-dispatcherServlet.xml</param-value>

    ????</init-param>

    ????<load-on-startup>1</load-on-startup>

    </servlet>

    ????????

    <servlet-mapping>

    ????<servlet-name>dispatcher</servlet-name>

    ????<url-pattern>/</url-pattern>

    </servlet-mapping>



    在這種情況下,DispatcherServlet和ContextLoaderListener會(huì)分別構(gòu)建不同作用范圍的容器(WebApplicationContext)。我們可以引入兩個(gè)不同的概念來(lái)對(duì)其進(jìn)行表述:ContextLoaderListener所初始化的容器,我們稱之為Root WebApplicationContext;而DispatcherServlet所初始化的容器,是SpringMVC WebApplicationContext

    同樣采取日志分析的方法,加入了ContextLoaderListener之后,整個(gè)啟動(dòng)日志變成了這樣:

    引用

    [main] INFO /sample - Initializing Spring root WebApplicationContext
    14:56:42,261? INFO ContextLoader:272 - Root WebApplicationContext: initialization started
    14:56:42,343 DEBUG StandardServletEnvironment:100 - Initializing new StandardServletEnvironment
    14:56:42,365? INFO XmlWebApplicationContext:495 - Refreshing Root WebApplicationContext: startup date [Mon Feb 06 14:56:42 CST 2012]; root of context hierarchy
    14:56:42,441 DEBUG PathMatchingResourcePatternResolver:550 - Looking for matching resources in directory tree [D:\Work\Demo2do\Sample\target\classes\context]
    14:56:42,442 DEBUG PathMatchingResourcePatternResolver:612 - Searching directory [D:\Work\Demo2do\Sample\target\classes\context] for files matching pattern [D:/Work/Demo2do/Sample/target/classes/context/applicationContext-*.xml]
    14:56:42,446 DEBUG PathMatchingResourcePatternResolver:351 - Resolved location pattern [classpath:context/applicationContext-*.xml] to resources [file [D:\Work\Demo2do\Sample\target\classes\context\applicationContext-configuration.xml]]
    14:56:42,447? INFO XmlBeanDefinitionReader:315 - Loading XML bean definitions from file [D:\Work\Demo2do\Sample\target\classes\context\applicationContext-configuration.xml]
    14:56:42,486 DEBUG PluggableSchemaResolver:140 - Loading schema mappings from [META-INF/spring.schemas]
    14:56:42,597 DEBUG DefaultBeanDefinitionDocumentReader:108 - Loading bean definitions
    14:56:42,658 DEBUG PathMatchingResourcePatternResolver:550 - Looking for matching resources in directory tree [D:\Work\Demo2do\Sample\target\classes\com\demo2do\sample]
    14:56:42,699 DEBUG ClassPathBeanDefinitionScanner:243 - Identified candidate component class: file [D:\Work\Demo2do\Sample\target\classes\com\demo2do\sample\service\impl\BlogServiceImpl.class]
    14:56:42,750 DEBUG XmlBeanDefinitionReader:216 - Loaded 5 bean definitions from location pattern [classpath:context/applicationContext-*.xml]
    14:56:42,750 DEBUG XmlWebApplicationContext:525 - Bean factory for Root WebApplicationContext: org.springframework.beans.factory.support.DefaultListableBeanFactory@478e4327: defining beans [blogService,org.springframework.context.annotation.internalConfigurationAnnotationProcessor,org.springframework.context.annotation.internalAutowiredAnnotationProcessor,org.springframework.context.annotation.internalRequiredAnnotationProcessor,org.springframework.context.annotation.internalCommonAnnotationProcessor]; root of factory hierarchy
    14:56:42,860 DEBUG ContextLoader:296 - Published root WebApplicationContext as ServletContext attribute with name [org.springframework.web.context.WebApplicationContext.ROOT]
    14:56:42,860? INFO ContextLoader:301 - Root WebApplicationContext: initialization completed in 596 ms


    14:56:42,935 DEBUG DispatcherServlet:115 - Initializing servlet 'dispatcher'
    14:56:42,974? INFO DispatcherServlet:444 - FrameworkServlet 'dispatcher': initialization started

    14:56:42,974 DEBUG DispatcherServlet:572 - Servlet with name 'dispatcher' will try to create custom WebApplicationContext context of class 'org.springframework.web.context.support.XmlWebApplicationContext', using parent context [Root WebApplicationContext: startup date [Mon Feb 06 14:56:42 CST 2012]; root of context hierarchy]
    14:56:42,979? INFO XmlWebApplicationContext:495 - Refreshing WebApplicationContext for namespace 'dispatcher-servlet': startup date [Mon Feb 06 14:56:42 CST 2012]; parent: Root WebApplicationContext
    14:56:42,983? INFO XmlBeanDefinitionReader:315 - Loading XML bean definitions from class path resource [web/applicationContext-dispatcherServlet.xml]
    14:56:42,987 DEBUG PluggableSchemaResolver:140 - Loading schema mappings from [META-INF/spring.schemas]
    14:56:43,035 DEBUG DefaultBeanDefinitionDocumentReader:108 - Loading bean definitions
    14:56:43,075 DEBUG PathMatchingResourcePatternResolver:550 - Looking for matching resources in directory tree [D:\Work\Demo2do\Sample\target\classes\com\demo2do\sample\web\controller]
    14:56:43,075 DEBUG PathMatchingResourcePatternResolver:612 - Searching directory [D:\Work\Demo2do\Sample\target\classes\com\demo2do\sample\web\controller] for files matching pattern [D:/Work/Demo2do/Sample/target/classes/com/demo2do/sample/web/controller/**/*.class]
    14:56:43,077 DEBUG PathMatchingResourcePatternResolver:351 - Resolved location pattern [classpath*:com/demo2do/sample/web/controller/**/*.class] to resources [file [D:\Work\Demo2do\Sample\target\classes\com\demo2do\sample\web\controller\BlogController.class], file [D:\Work\Demo2do\Sample\target\classes\com\demo2do\sample\web\controller\UserController.class]]
    14:56:43,079 DEBUG ClassPathBeanDefinitionScanner:243 - Identified candidate component class: file [D:\Work\Demo2do\Sample\target\classes\com\demo2do\sample\web\controller\BlogController.class]
    14:56:43,080 DEBUG ClassPathBeanDefinitionScanner:243 - Identified candidate component class: file [D:\Work\Demo2do\Sample\target\classes\com\demo2do\sample\web\controller\UserController.class]
    14:56:43,089 DEBUG XmlBeanDefinitionReader:216 - Loaded 19 bean definitions from location pattern [classpath:web/applicationContext-dispatcherServlet.xml]
    14:56:43,089 DEBUG XmlWebApplicationContext:525 - Bean factory for WebApplicationContext for namespace 'dispatcher-servlet': org.springframework.beans.factory.support.DefaultListableBeanFactory@5e6458a6: defining beans [org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping#0,org.springframework.format.support.FormattingConversionServiceFactoryBean#0,org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter#0,org.springframework.web.servlet.handler.MappedInterceptor#0,org.springframework.web.servlet.mvc.method.annotation.ExceptionHandlerExceptionResolver#0,org.springframework.web.servlet.mvc.annotation.ResponseStatusExceptionResolver#0,org.springframework.web.servlet.mvc.support.DefaultHandlerExceptionResolver#0,org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping,org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter,org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter,blogController,userController,org.springframework.context.annotation.internalConfigurationAnnotationProcessor,org.springframework.context.annotation.internalAutowiredAnnotationProcessor,org.springframework.context.annotation.internalRequiredAnnotationProcessor,org.springframework.context.annotation.internalCommonAnnotationProcessor,org.springframework.web.servlet.resource.ResourceHttpRequestHandler#0,org.springframework.web.servlet.handler.SimpleUrlHandlerMapping#0,org.springframework.web.servlet.view.InternalResourceViewResolver#0]; parent: org.springframework.beans.factory.support.DefaultListableBeanFactory@478e4327
    14:56:43,323 DEBUG RequestMappingHandlerMapping:98 - Looking for request mappings in application context: WebApplicationContext for namespace 'dispatcher-servlet': startup date [Mon Feb 06 14:56:42 CST 2012]; parent: Root WebApplicationContext
    14:56:43,345? INFO RequestMappingHandlerMapping:188 - Mapped "{[/blog],methods=[],params=[],headers=[],consumes=[],produces=[],custom=[]}" onto public org.springframework.web.servlet.ModelAndView com.demo2do.sample.web.controller.BlogController.index()
    14:56:43,346? INFO RequestMappingHandlerMapping:188 - Mapped "{[/login],methods=[],params=[],headers=[],consumes=[],produces=[],custom=[]}" onto public org.springframework.web.servlet.ModelAndView com.demo2do.sample.web.controller.UserController.login(java.lang.String,java.lang.String)
    14:56:43,707 DEBUG BeanNameUrlHandlerMapping:71 - Looking for URL mappings in application context: WebApplicationContext for namespace 'dispatcher-servlet': startup date [Mon Feb 06 14:56:42 CST 2012]; parent: Root WebApplicationContext
    14:56:43,828? INFO SimpleUrlHandlerMapping:314 - Mapped URL path [/static/**] onto handler 'org.springframework.web.servlet.resource.ResourceHttpRequestHandler#0'
    14:56:43,883 DEBUG DispatcherServlet:523 - Published WebApplicationContext of servlet 'dispatcher' as ServletContext attribute with name [org.springframework.web.servlet.FrameworkServlet.CONTEXT.dispatcher]
    14:56:43,883? INFO DispatcherServlet:463 - FrameworkServlet 'dispatcher': initialization completed in 909 ms
    14:56:43,883 DEBUG DispatcherServlet:136 - Servlet 'dispatcher' configured successfully

    2687 [main] INFO org.mortbay.log - Started SelectChannelConnector@0.0.0.0:8080
    Jetty Server started, use 2901 ms



    整個(gè)啟動(dòng)日志被我們分為了2段。第一段的過(guò)程初始化的是Root WebApplicationContext;而第二段的過(guò)程初始化的是SpringMVC的WebApplicationContext。我們還是使用了紅色的標(biāo)記和藍(lán)色標(biāo)記指出了在整個(gè)初始化過(guò)程中的一些重要事件。其中,有這樣一段內(nèi)容值得我們注意:

    引用

    14:56:42,979? INFO XmlWebApplicationContext:495 - Refreshing WebApplicationContext for namespace 'dispatcher-servlet': startup date [Mon Feb 06 14:56:42 CST 2012]; parent: Root WebApplicationContext



    在這段日志中,非常明確地指出了SpringMVC WebApplicationContext與Root WebApplicationContext之間的關(guān)系:從屬關(guān)系。因?yàn)楦鶕?jù)這段日志的表述,SpringMVC WebApplicationContext能夠感知到Root WebApplicationContext的存在,并且將其作為parent容器。

    Spring正是使用這種Parent-Child的容器關(guān)系來(lái)對(duì)不同的編程層次進(jìn)行劃分。這種我們俗稱的父子關(guān)系實(shí)際上不僅僅是一種從屬關(guān)系,更是一種引用關(guān)系。從剛才的日志分析中,我們可以看出:SpringMVC中所定義的一切組件能夠無(wú)縫地與Root WebApplicationContext中的組件整合。

    到此為止,我們針對(duì)圖中以web.xml為核心的箭頭分支進(jìn)行了講解,讀者可以將圖中的內(nèi)容與上面的文字說(shuō)明對(duì)照再次加以理解。

    【組件默認(rèn)行為的指定】

    DispatcherServlet的初始化主線的執(zhí)行體系是順著其繼承結(jié)構(gòu)依次進(jìn)行的,我們?cè)谥霸?jīng)討論過(guò)它的執(zhí)行次序。所以,只有在FrameworkServlet完成了對(duì)于WebApplicationContext和組件的初始化之后,執(zhí)行權(quán)才被正式轉(zhuǎn)移到DispatcherServlet中。我們可以來(lái)看看DispatcherServlet此時(shí)究竟干了哪些事:

    Java代碼 ?

  • /**?
  • ?*?This?implementation?calls?{@link?#initStrategies}.?
  • ?*/??
  • @Override??
  • protected?void?onRefresh(ApplicationContext?context)?{??
  • ????initStrategies(context);??
  • }??
  • ??
  • /**?
  • ?*?Initialize?the?strategy?objects?that?this?servlet?uses.?
  • ?*?<p>May?be?overridden?in?subclasses?in?order?to?initialize?further?strategy?objects.?
  • ?*/??
  • protected?void?initStrategies(ApplicationContext?context)?{??
  • ????initMultipartResolver(context);??
  • ????initLocaleResolver(context);??
  • ????initThemeResolver(context);??
  • ????initHandlerMappings(context);??
  • ????initHandlerAdapters(context);??
  • ????initHandlerExceptionResolvers(context);??
  • ????initRequestToViewNameTranslator(context);??
  • ????initViewResolvers(context);??
  • ????initFlashMapManager(context);??
  • }??
  • /**

    * This implementation calls {@link #initStrategies}.

    */

    @Override

    protected void onRefresh(ApplicationContext context) {

    ????initStrategies(context);

    }

    ?

    /**

    * Initialize the strategy objects that this servlet uses.

    * <p>May be overridden in subclasses in order to initialize further strategy objects.

    */

    protected void initStrategies(ApplicationContext context) {

    ????initMultipartResolver(context);

    ????initLocaleResolver(context);

    ????initThemeResolver(context);

    ????initHandlerMappings(context);

    ????initHandlerAdapters(context);

    ????initHandlerExceptionResolvers(context);

    ????initRequestToViewNameTranslator(context);

    ????initViewResolvers(context);

    ????initFlashMapManager(context);

    }



    onRefresh是FrameworkServlet中預(yù)留的擴(kuò)展方法,在DispatcherServlet中做了一個(gè)基本實(shí)現(xiàn):initStrategies。我們粗略一看,很容易就能明白DispatcherServlet到底在這里干些什么了:初始化組件

    讀者或許會(huì)問(wèn),組件不是已經(jīng)在WebApplicationContext初始化的時(shí)候已經(jīng)被初始化過(guò)了嘛?這里所謂的組件初始化,指的又是什么呢?讓我們來(lái)看看其中的一個(gè)方法的源碼:

    Java代碼 ?

  • /**?
  • ?*?Initialize?the?MultipartResolver?used?by?this?class.?
  • ?*?<p>If?no?bean?is?defined?with?the?given?name?in?the?BeanFactory?for?this?namespace,?
  • ?*?no?multipart?handling?is?provided.?
  • ?*/??
  • private?void?initMultipartResolver(ApplicationContext?context)?{??
  • ????try?{??
  • ????????this.multipartResolver?=?context.getBean(MULTIPART_RESOLVER_BEAN_NAME,?MultipartResolver.class);??
  • ????????if?(logger.isDebugEnabled())?{??
  • ????????????logger.debug("Using?MultipartResolver?["?+?this.multipartResolver?+?"]");??
  • ????????}??
  • ????}?catch?(NoSuchBeanDefinitionException?ex)?{??
  • ????????//?Default?is?no?multipart?resolver.??
  • ????????this.multipartResolver?=?null;??
  • ????????if?(logger.isDebugEnabled())?{??
  • ????????????logger.debug("Unable?to?locate?MultipartResolver?with?name?'"?+?MULTIPART_RESOLVER_BEAN_NAME?+??
  • ????????????????????"':?no?multipart?request?handling?provided");??
  • ????????}??
  • ????}??
  • }??
  • /**

    * Initialize the MultipartResolver used by this class.

    * <p>If no bean is defined with the given name in the BeanFactory for this namespace,

    * no multipart handling is provided.

    */

    private void initMultipartResolver(ApplicationContext context) {

    ????try {

    ????????this.multipartResolver = context.getBean(MULTIPART_RESOLVER_BEAN_NAME, MultipartResolver.class);

    ????????if (logger.isDebugEnabled()) {

    ????????????logger.debug("Using MultipartResolver [" + this.multipartResolver + "]");

    ????????}

    ????} catch (NoSuchBeanDefinitionException ex) {

    ????????// Default is no multipart resolver.

    ????????this.multipartResolver = null;

    ????????if (logger.isDebugEnabled()) {

    ????????????logger.debug("Unable to locate MultipartResolver with name '" + MULTIPART_RESOLVER_BEAN_NAME +

    ????????????????????"': no multipart request handling provided");

    ????????}

    ????}

    }



    原來(lái),這里的初始化,指的是DispatcherServlet從容器(WebApplicationContext)中讀取組件的實(shí)現(xiàn)類,并緩存于DispatcherServlet內(nèi)部的過(guò)程。還記得我們之前給出的DispatcherServlet的數(shù)據(jù)結(jié)構(gòu)嗎?這些位于DispatcherServlet內(nèi)部的組件實(shí)際上只是一些來(lái)源于容器緩存實(shí)例,不過(guò)它們同樣也是DispatcherServlet進(jìn)行后續(xù)操作的基礎(chǔ)。

    :我們?cè)诘谝黄恼轮芯驮?jīng)提到過(guò)Servlet實(shí)例內(nèi)部的屬性的訪問(wèn)有線程安全問(wèn)題。而在這里,我們可以看到所有的組件都以Servlet內(nèi)部屬性的形式被調(diào)用,充分證實(shí)了這些組件本身也都是無(wú)狀態(tài)的單例對(duì)象,所以我們?cè)谶@里不必考慮線程安全的問(wèn)題。

    如果對(duì)上面的代碼加以詳細(xì)分析,我們會(huì)發(fā)現(xiàn)initMultipartResolver的過(guò)程是查找特定MultipartResolver實(shí)現(xiàn)類的過(guò)程。因?yàn)樵谌萜髦胁檎医M件的時(shí)候,采取的是根據(jù)特定名稱(MULTIPART_RESOLVER_BEAN_NAME)進(jìn)行查找的策略。由此,我們可以看到DispatcherServlet進(jìn)行組件初始化的特點(diǎn):

    downpour 寫道

    結(jié)論 DispatcherServlet中對(duì)于組件的初始化過(guò)程實(shí)際上是應(yīng)用程序在WebApplicationContext中選擇和查找組件實(shí)現(xiàn)類的過(guò)程,也是指定組件在SpringMVC中的默認(rèn)行為方式的過(guò)程。



    除了根據(jù)特定名稱進(jìn)行查找的策略以外,我們還對(duì)DispatcherServlet中指定SpringMVC默認(rèn)行為方式的其他的策略進(jìn)行的總結(jié):

    • 名稱查找 —— 根據(jù)bean的名字在容器中查找相應(yīng)的實(shí)現(xiàn)類
    • 自動(dòng)搜索 —— 自動(dòng)搜索容器中所有某個(gè)特定組件(接口)的所有實(shí)現(xiàn)類
    • 默認(rèn)配置 —— 根據(jù)一個(gè)默認(rèn)的配置文件指定進(jìn)行實(shí)現(xiàn)類加載

    這三條策略恰巧在initHandlerMappings的過(guò)程中都有體現(xiàn),讀者可以從其源碼中找到相應(yīng)的線索:

    Java代碼 ?

  • private?void?initHandlerAdapters(ApplicationContext?context)?{??
  • ????this.handlerAdapters?=?null;??
  • ??
  • ????if?(this.detectAllHandlerAdapters)?{??
  • ????????//?Find?all?HandlerAdapters?in?the?ApplicationContext,?including?ancestor?contexts.??
  • ????????Map<String,?HandlerAdapter>?matchingBeans?=?BeanFactoryUtils.beansOfTypeIncludingAncestors(context,?HandlerAdapter.class,?true,?false);??
  • ????????if?(!matchingBeans.isEmpty())?{??
  • ????????????this.handlerAdapters?=?new?ArrayList<HandlerAdapter>(matchingBeans.values());??
  • ????????????//?We?keep?HandlerAdapters?in?sorted?order.??
  • ????????????OrderComparator.sort(this.handlerAdapters);??
  • ????????}??
  • ????}??
  • ????else?{??
  • ????????try?{??
  • ????????????HandlerAdapter?ha?=?context.getBean(HANDLER_ADAPTER_BEAN_NAME,?HandlerAdapter.class);??
  • ????????????this.handlerAdapters?=?Collections.singletonList(ha);??
  • ????????}??
  • ????????catch?(NoSuchBeanDefinitionException?ex)?{??
  • ????????????//?Ignore,?we'll?add?a?default?HandlerAdapter?later.??
  • ????????}??
  • ????}??
  • ??
  • ????//?Ensure?we?have?at?least?some?HandlerAdapters,?by?registering??
  • ????//?default?HandlerAdapters?if?no?other?adapters?are?found.??
  • ????if?(this.handlerAdapters?==?null)?{??
  • ????????this.handlerAdapters?=?getDefaultStrategies(context,?HandlerAdapter.class);??
  • ????????if?(logger.isDebugEnabled())?{??
  • ????????????logger.debug("No?HandlerAdapters?found?in?servlet?'"?+?getServletName()?+?"':?using?default");??
  • ????????}??
  • ????}??
  • }??
  • private void initHandlerAdapters(ApplicationContext context) {

    ????this.handlerAdapters = null;

    ?

    ????if (this.detectAllHandlerAdapters) {

    ????????// Find all HandlerAdapters in the ApplicationContext, including ancestor contexts.

    ????????Map<String, HandlerAdapter> matchingBeans = BeanFactoryUtils.beansOfTypeIncludingAncestors(context, HandlerAdapter.class, true, false);

    ????????if (!matchingBeans.isEmpty()) {

    ????????????this.handlerAdapters = new ArrayList<HandlerAdapter>(matchingBeans.values());

    ????????????// We keep HandlerAdapters in sorted order.

    ????????????OrderComparator.sort(this.handlerAdapters);

    ????????}

    ????}

    ????else {

    ????????try {

    ????????????HandlerAdapter ha = context.getBean(HANDLER_ADAPTER_BEAN_NAME, HandlerAdapter.class);

    ????????????this.handlerAdapters = Collections.singletonList(ha);

    ????????}

    ????????catch (NoSuchBeanDefinitionException ex) {

    ????????????// Ignore, we'll add a default HandlerAdapter later.

    ????????}

    ????}

    ?

    ????// Ensure we have at least some HandlerAdapters, by registering

    ????// default HandlerAdapters if no other adapters are found.

    ????if (this.handlerAdapters == null) {

    ????????this.handlerAdapters = getDefaultStrategies(context, HandlerAdapter.class);

    ????????if (logger.isDebugEnabled()) {

    ????????????logger.debug("No HandlerAdapters found in servlet '" + getServletName() + "': using default");

    ????????}

    ????}

    }



    這里有必要對(duì)"默認(rèn)策略"做一個(gè)簡(jiǎn)要的說(shuō)明。SpringMVC為一些核心組件設(shè)置了默認(rèn)行為方式的說(shuō)明,這個(gè)說(shuō)明以一個(gè)properties文件的形式位于SpringMVC分發(fā)包(例如spring-webmvc-3.1.0.RELEASE.jar)的內(nèi)部:



    我們可以觀察一下DispatcherServlet.properties的內(nèi)容:

    引用

    # Default implementation classes for DispatcherServlet's strategy interfaces.
    # Used as fallback when no matching beans are found in the DispatcherServlet context.
    # Not meant to be customized by application developers.

    org.springframework.web.servlet.LocaleResolver=org.springframework.web.servlet.i18n.AcceptHeaderLocaleResolver

    org.springframework.web.servlet.ThemeResolver=org.springframework.web.servlet.theme.FixedThemeResolver

    org.springframework.web.servlet.HandlerMapping=org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping,\
    org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping

    org.springframework.web.servlet.HandlerAdapter=org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter,\
    org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter,\
    org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter

    org.springframework.web.servlet.HandlerExceptionResolver=org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerExceptionResolver,\
    org.springframework.web.servlet.mvc.annotation.ResponseStatusExceptionResolver,\
    org.springframework.web.servlet.mvc.support.DefaultHandlerExceptionResolver

    org.springframework.web.servlet.RequestToViewNameTranslator=org.springframework.web.servlet.view.DefaultRequestToViewNameTranslator

    org.springframework.web.servlet.ViewResolver=org.springframework.web.servlet.view.InternalResourceViewResolver

    org.springframework.web.servlet.FlashMapManager=org.springframework.web.servlet.support.DefaultFlashMapManager



    結(jié)合剛才initHandlerMappings的源碼,我們可以發(fā)現(xiàn)如果沒(méi)有開啟detectAllHandlerAdapters選項(xiàng)或者根據(jù)HANDLER_ADAPTER_BEAN_NAME的名稱沒(méi)有找到相應(yīng)的組件實(shí)現(xiàn)類,就會(huì)使用DispatcherServlet.properties文件中對(duì)于HandlerMapping接口的實(shí)現(xiàn)來(lái)進(jìn)行組件默認(rèn)行為的初始化。

    由此可見,DispatcherServlet.properties中所指定的所有接口的實(shí)現(xiàn)方式在Spring的容器WebApplicationContext中總有相應(yīng)的定義。這一點(diǎn),我們?cè)诮M件的討論中還會(huì)詳談。

    這個(gè)部分我們的側(cè)重點(diǎn)是圖中DispatcherServlet與容器之間的關(guān)系。讀者需要理解的是圖中為什么會(huì)有兩份組件定義,它們之間的區(qū)別在哪里,以及DispatcherServlet在容器中查找組件的三種策略。

    小結(jié)

    在本文中,我們對(duì)SpringMVC的核心類:DispatcherServlet進(jìn)行了一番梳理。也對(duì)整個(gè)SpringMVC的兩條主線之一的初始化主線做了詳細(xì)的分析。

    對(duì)于DispatcherServlet而言,重要的其實(shí)并不是這個(gè)類中的代碼和邏輯,而是應(yīng)該掌握這個(gè)類在整個(gè)框架中的作用以及與SpringMVC中其他要素的關(guān)系。

    對(duì)于初始化主線而言,核心其實(shí)僅僅在于那張筆者為大家精心打造的圖。讀者只要掌握了這張圖,相信對(duì)整個(gè)SpringMVC的初始化過(guò)程會(huì)有一個(gè)全新的認(rèn)識(shí)。

    轉(zhuǎn)載于:https://www.cnblogs.com/xiangxs/p/5052025.html

    總結(jié)

    以上是生活随笔為你收集整理的SpringMVC深度探险(三) —— DispatcherServlet与初始化主线的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

    如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。