您好,登錄后才能下訂單哦!
這篇文章給大家介紹如何進(jìn)行Spring源代碼解析IoC容器在Web容器中的啟動,內(nèi)容非常詳細(xì),感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助。
在web容器中,通過ServletContext為Spring的IOC容器提供宿主環(huán)境,對應(yīng)的建立起一個IOC容器的體系。其中,首先需要建立的是根上下文,這個上下文持有的對象可以有業(yè)務(wù)對象,數(shù)據(jù)存取對象,資源,事物管理器等各種中間層對象。在這個上下文的基礎(chǔ)上,和web MVC相關(guān)還會有一個上下文來保存控制器之類的MVC對象,這樣就構(gòu)成了一個層次化的上下文結(jié)構(gòu)。在web容器中啟動Spring應(yīng)用程序就是一個建立這個上下文體系的過程。Spring為web應(yīng)用提供了上下文的擴展接口
WebApplicationContext:
Java代碼
而一般的啟動過程,Spring會使用一個默認(rèn)的實現(xiàn),XmlWebApplicationContext - 這個上下文實現(xiàn)作為在web容器中的根上下文容器被建立起來,具體的建立過程在下面我們會詳細(xì)分析。
Java代碼
對于一個Spring激活的web應(yīng)用程序,可以通過使用Spring代碼聲明式的指定在web應(yīng)用程序啟動時載入應(yīng)用程序上下文(WebApplicationContext),Spring的ContextLoader是提供這樣性能的類,我們可以使用 ContextLoaderServlet或者ContextLoaderListener的啟動時載入的Servlet來實例化Spring IOC容器 - 為什么會有兩個不同的類來裝載它呢,這是因為它們的使用需要區(qū)別不同的Servlet容器支持的Serlvet版本。但不管是 ContextLoaderSevlet還是 ContextLoaderListener都使用ContextLoader來完成實際的WebApplicationContext的初始化工作。這個ContextLoder就像是Spring Web應(yīng)用程序在Web容器中的加載器booter。當(dāng)然這些Servlet的具體使用我們都要借助web容器中的部署描述符來進(jìn)行相關(guān)的定義。
下面我們使用ContextLoaderListener作為載入器作一個詳細(xì)的分析,這個Servlet的監(jiān)聽器是根上下文被載入的地方,也是整個 Spring web應(yīng)用加載上下文的第一個地方;從加載過程我們可以看到,首先從Servlet事件中得到ServletContext,然后可以讀到配置好的在web.xml的中的各個屬性值,然后ContextLoder實例化WebApplicationContext并完成其載入和初始化作為根上下文。當(dāng)這個根上下文被載入后,它被綁定到web應(yīng)用程序的ServletContext上。任何需要訪問該ApplicationContext的應(yīng)用程序代碼都可以從WebApplicationContextUtils類的靜態(tài)方法來得到:
Java代碼
WebApplicationContext getWebApplicationContext(ServletContext sc)
以Tomcat作為Servlet容器為例,下面是具體的步驟:
1.Tomcat 啟動時需要從web.xml中讀取啟動參數(shù),在web.xml中我們需要對ContextLoaderListener進(jìn)行配置,對于在web應(yīng)用啟動入口是在ContextLoaderListener中的初始化部分;從Spring MVC上看,實際上在web容器中維護(hù)了一系列的IOC容器,其中在ContextLoader中載入的IOC容器作為根上下文而存在于 ServletContext中。
Java代碼
通過ContextLoader建立起根上下文的過程,我們可以在ContextLoader中看到:
Java代碼
建立根上下文的父上下文使用的是下面的代碼,取決于在web.xml中定義的參數(shù):locatorFactorySelector,這是一個可選參數(shù):
Java代碼
得到根上下文的父上下文以后,就是根上下文的創(chuàng)建過程:
Java代碼
初始化根ApplicationContext后將其存儲到SevletContext中去以后,這樣就建立了一個全局的關(guān)于整個應(yīng)用的上下文。這個根上下文會被以后的DispatcherServlet初始化自己的時候作為自己ApplicationContext的父上下文。這個在對 DispatcherServlet做分析的時候我們可以看看到。
3.完成對ContextLoaderListener的初始化以后, Tomcat開始初始化DispatchServlet,- 還記得我們在web.xml中隊載入次序進(jìn)行了定義。DispatcherServlet會建立自己的ApplicationContext,同時建立這個自己的上下文的時候會從ServletContext中得到根上下文作為父上下文,然后再對自己的上下文進(jìn)行初始化,并最后存到 ServletContext中去供以后檢索和使用。
可以從DispatchServlet的父類FrameworkServlet的代碼中看到大致的初始化過程,整個ApplicationContext的創(chuàng)建過程和ContextLoder創(chuàng)建的過程相類似:
Java代碼
對initWebApplicationContext()調(diào)用的代碼如下:
Java代碼
其中我們看到調(diào)用了WebApplicationContextUtils的靜態(tài)方法得到根ApplicationContext:
Java代碼
4. 然后就是DispatchServlet中對Spring MVC的配置過程,首先對配置文件中的定義元素進(jìn)行配置 - 請注意這個時候我們的WebApplicationContext已經(jīng)建立起來了,也意味著DispatcherServlet有自己的定義資源,可以需要從web.xml中讀取bean的配置信息,通常我們會使用單獨的xml文件來配置MVC中各個要素定義,這里和web容器相關(guān)的加載過程實際上已經(jīng)完成了,下面的處理和普通的Spring應(yīng)用程序的編寫沒有什么太大的差別,我們先看看MVC的初始化過程:
Java代碼
5. 這樣MVC的框架就建立起來了,DispatchServlet對接受到的HTTP Request進(jìn)行分發(fā)處理由doService()完成,具體的MVC處理過程我們在doDispatch()中完成,其中包括使用Command模式建立執(zhí)行鏈,顯示模型數(shù)據(jù)等,這些處理我們都可以在DispatcherServlet的代碼中看到:
Java代碼
實際的請求分發(fā)由doDispatch(request,response)來完成:
Java代碼
這樣具體的MVC模型的實現(xiàn)就由bean配置文件里定義好的view resolver,handler這些類來實現(xiàn)用戶代碼的功能。
總結(jié)上面的過程,我們看到在web容器中,ServletContext可以持有一系列的web上下文,而在整個web上下文中存在一個根上下文來作為其它 Servlet上下文的父上下文。這個根上下文是由ContextLoader載入并進(jìn)行初始化的,對于我們的web應(yīng)用, DispatcherSerlvet載入并初始化自己的上下文,這個上下文的父上下文是根上下文,并且我們也能從ServletContext中根據(jù) Servlet的名字來檢索到我們需要的對應(yīng)于這個Servlet的上下文,但是根上下文的名字是由Spring唯一確定的。這個 DispactcherServlet建立的上下文就是我們開發(fā)Spring MVC應(yīng)用的IOC容器。
關(guān)于如何進(jìn)行Spring源代碼解析IoC容器在Web容器中的啟動就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。