Pogy原創(chuàng),未經(jīng)允許,請(qǐng)勿轉(zhuǎn)載?。?!
創(chuàng)新互聯(lián)建站一直通過網(wǎng)站建設(shè)和網(wǎng)站營(yíng)銷幫助企業(yè)獲得更多客戶資源。 以"深度挖掘,量身打造,注重實(shí)效"的一站式服務(wù),以網(wǎng)站制作、網(wǎng)站建設(shè)、移動(dòng)互聯(lián)產(chǎn)品、成都營(yíng)銷網(wǎng)站建設(shè)服務(wù)為核心業(yè)務(wù)。10年網(wǎng)站制作的經(jīng)驗(yàn),使用新網(wǎng)站建設(shè)技術(shù),全新開發(fā)出的標(biāo)準(zhǔn)網(wǎng)站,不但價(jià)格便宜而且實(shí)用、靈活,特別適合中小公司網(wǎng)站制作。網(wǎng)站管理系統(tǒng)簡(jiǎn)單易用,維護(hù)方便,您可以完全操作網(wǎng)站資料,是中小公司快速網(wǎng)站建設(shè)的選擇。springMVC是一個(gè)建立在spring框架基礎(chǔ)之上的一個(gè)web層框架,它負(fù)責(zé)發(fā)送每個(gè)請(qǐng)求到合適的處理程序,使用視圖來最終返回響應(yīng)結(jié)果的概念。
SpringMVC最重要的一個(gè)類是DispatchServlet這個(gè)類,當(dāng)我們打開這個(gè)源碼,感到一臉蒙蔽。但是坐下來冷靜一下看看,發(fā)現(xiàn),這個(gè)東西竟然繼承了一些類
跟過去看看
好像也沒啥線索,繼續(xù)跟
這時(shí)有了重大發(fā)現(xiàn),發(fā)現(xiàn)這個(gè)DispatchServlet終歸結(jié)底還是繼承了HttpServlet,那么我們就可以按照普通的servlet的生命周期來分析它了。
Servlet的生命周期大家都還記得吧,不記得沒關(guān)系,我來簡(jiǎn)單列一下
1. 首先tomcat啟動(dòng)的時(shí)候會(huì)去查詢web.xml中的配置文件找到servlet的配置,然后會(huì)通過new關(guān)鍵字創(chuàng)建這個(gè)servlet的實(shí)例。
2. 然后調(diào)用這個(gè)servlet的init方法去初始化,然后把這個(gè)servlet放入容器中管理,等待使用。
3. 當(dāng)容器關(guān)閉時(shí),tomcat會(huì)調(diào)用這個(gè)servlet的destroy方法來做一些善后工作。
Ok,下面我們就找init方法吧,看看這里面都干了些什么吧
這里面沒有什么太復(fù)雜的邏輯,就是把配置DispatchServlet配置參數(shù)加載到內(nèi)存中,以供spring使用。有的朋友就問了,什么參數(shù)
主要就是這里面調(diào)用了initServletBean方法,這個(gè)方法在這個(gè)類中是空的,而它的子類實(shí)現(xiàn)了它,springmvc中用了很多類似的設(shè)計(jì)模式,叫模板方法模式。簡(jiǎn)單講就是在父類中寫好,你要調(diào)用哪些方法,我在子類中去實(shí)現(xiàn),這樣就按照了父類的邏輯順序調(diào)用了相應(yīng)的方法,而且子類可以有不同的實(shí)現(xiàn)。
好,下面我么就找哪個(gè)子類實(shí)現(xiàn)了這個(gè)initServletBean方法吧,跟我走。。。
看看這個(gè)HttpServletBean的繼承體系如下
先找FrameworkServlet
我們尋尋覓覓,果然沒猜錯(cuò),方法的實(shí)現(xiàn),寫在了子類,而且還不少,我來分析下吧
略過日志記錄往下走
調(diào)用了當(dāng)前類中寫的一個(gè)方法initWebApplicationContext方法,這個(gè)方法就很重要了,坐下來,喝杯茶,慢慢品嘗spring這道美味可口的大餐。
先把源碼粘上來,
第一步spring通過工具類獲取父級(jí)WebApplicationContext容器,有的朋友問了,這個(gè)父級(jí)容器哪來的,何時(shí)創(chuàng)建的。
就是在這個(gè)監(jiān)聽器運(yùn)行的時(shí)候創(chuàng)建的,我們知道web服務(wù)器啟動(dòng)的時(shí)候,3大組件的創(chuàng)建順序是Listener>Filter >Servlet。
所以這里是可以獲取到父級(jí)容器的。當(dāng)然也可以獲取不到這種情況就不討論了,因?yàn)?,不可能說你用springMVC而不用spring框架的。
這里有幾個(gè)if判斷,我已經(jīng)標(biāo)號(hào)了,給大家分析一下執(zhí)行流。
首先webApplicationContext一定是空的,為什么,因?yàn)槟銊倖?dòng),肯定是空的。所以第一個(gè)if暫時(shí)不會(huì)執(zhí)行。進(jìn)入到第二個(gè)if了,這里調(diào)用了findWebApplicationContext,其實(shí)這里也會(huì)返回一個(gè)null,從這個(gè)findwebApplicationContext名稱就知道,它只查找并不創(chuàng)建。把源碼附上,大家看看
因此wac還是空的,所以進(jìn)入第三個(gè)if語句,我們進(jìn)到createWabApplicationContext方法中一探究竟。
這里的1號(hào)代碼塊中是日志,2號(hào)代碼塊中式判斷的,3號(hào)代碼塊是創(chuàng)建一個(gè)可配置的web應(yīng)用上下文,也就是web容器,4號(hào)代碼塊是初始化一下這個(gè)容器。比如設(shè)置之前的rootContext為父級(jí)容器。
最后把這個(gè)創(chuàng)建的容器返回,至此,我們的springMVC容器就這樣誕生了。但是還沒結(jié)束,在Framework的initWebApplicationContext()方法里還有if,我把它粘上來
接著之前的if序號(hào)編號(hào)
看看這個(gè)字段的初始值
所以這個(gè)onRefresh()方法是要執(zhí)行的,跟進(jìn)去發(fā)現(xiàn)空實(shí)現(xiàn),意料之內(nèi)的事啦,沒關(guān)系的,模板方法模式嘛,找子類,有線索,
這個(gè)方法又調(diào)用了一個(gè)initStrategies(。。)方法,這個(gè)方法干了什么勾當(dāng) ,跟進(jìn)去看看
這里我直接加上注釋了
這里我就只跟一個(gè)進(jìn)去看看吧,initMultipartResolver(context),進(jìn)去看看
注意我畫線的這個(gè)段代碼,大家考慮有沒有問題啊,仔細(xì)思考哦
當(dāng)然有問題,我們又沒有在容器中注入這個(gè)解析器,它怎么能get到呢,出鬼了。
嘻嘻
另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無理由+7*72小時(shí)售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國(guó)服務(wù)器、虛擬主機(jī)、免備案服務(wù)器”等云主機(jī)租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡(jiǎn)單易用、服務(wù)可用性高、性價(jià)比高”等特點(diǎn)與優(yōu)勢(shì),專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場(chǎng)景需求。