【聊透SpringMVC】SpringMVC“傳統(tǒng)”方式的啟動(dòng)過程
打成war包并放入Tomcat等Servlet容器下面運(yùn)行的,都認(rèn)為是SpringMVC傳統(tǒng)的啟動(dòng)方式。
和SpringBoot連用且采用內(nèi)嵌Web服務(wù)器并打成jar包直接運(yùn)行的,可以認(rèn)為是SpringMVC現(xiàn)代的啟動(dòng)方式。
傳統(tǒng)的啟動(dòng)過程
基于web.xml(配置文件)的方式啟動(dòng)肯定算傳統(tǒng)的,但由于現(xiàn)在web.xml幾乎已經(jīng)絕跡,所以就不考慮它了。
與之相對(duì)的就是基于編程(寫代碼)的方式啟動(dòng),流行于前幾年的SSM(Spring、SpringMVC、MyBatis)中。當(dāng)然也算傳統(tǒng)的。
在上一篇中講到,通過一個(gè)“小橋式”的接口ServletContainerInitializer(Servlet容器初始化器)把Tomcat的啟動(dòng)和初始化進(jìn)程帶到了SpringMVC里。
在這個(gè)“橋式”接口上可以指定“感興趣”的類或接口,SpringMVC指定的是WebApplicationInitializer(Web應(yīng)用初始化器)接口,意圖已經(jīng)很明顯,就是通過這個(gè)初始化器接口來完成SpringMVC應(yīng)用的啟動(dòng)和初始化。
我們先來看下這個(gè)初始化接口,如下圖01:
它只有一個(gè)onStartup方法,方法只有一個(gè)參數(shù)就是ServletContext,這個(gè)ServletContext由Tomcat創(chuàng)建好后提供給SpringMVC,SpringMVC在啟動(dòng)過程中調(diào)用這個(gè)onStartup方法,在這個(gè)方法內(nèi)完成自身的創(chuàng)建和初始化,還要把Servlet和Filter等注冊(cè)到ServletContext里。
這些工作都是SpringMVC要做的,而不是我們要做的,所以SpringMVC肯定已經(jīng)實(shí)現(xiàn)了這個(gè)接口,我們查看下類型信息,如下圖02:
我們發(fā)現(xiàn)了一個(gè)看著很重要的類,就是:
AbstractAnnotationConfigDispatcherServletInitializer
可惜這個(gè)類是抽象的,肯定是不能直接用的,但是它里面已經(jīng)包含了剛剛上面提到的所有完整的啟動(dòng)邏輯過程。
如果你對(duì)SSM很熟悉或Spring的官方文檔看的很熟悉的話,你一定知道這個(gè)類是怎么用的。是的,我們需要定義一個(gè)類來繼承它即可。
先看下官方文檔上給的用法,如下圖03:
繼承之后,我們需要提供三方面信息,一個(gè)是用于注冊(cè)到根容器中的類,一個(gè)是用于注冊(cè)到Servlet容器中的類,一個(gè)是核心Servlet的映射URL。
注意,這里說的容器指的是Spring的ApplicationContext這個(gè)容器,其中根容器和Servlet容器是父子關(guān)系,且在SpringMVC中核心Servlet映射的URL必須是“/”。
下面給出一個(gè)我在幾年前為公司搭建框架時(shí)的代碼,如下圖04:
這就是以編程的方式來完成SpringMVC的啟動(dòng)。我們自己定義的這個(gè)類就是前文提到的“感興趣”的類。
這個(gè)類是不用(或不能)向Spring容器注冊(cè)的,因?yàn)檫@個(gè)類是感興趣的類,所以Tomcat會(huì)從jar包里把它找出來,這樣SpringMVC就拿到了我們定義的這個(gè)類。
其實(shí)最主要的是這時(shí)候根本還沒有Spring容器呢,哈哈,因?yàn)镾pring容器就是在這個(gè)類里才創(chuàng)建出來的,有點(diǎn)意思吧。
其實(shí)這個(gè)類里包含的內(nèi)容非常多,我們都可以使用寫代碼的方式來進(jìn)行配置。下面舉幾個(gè)示例。
比如對(duì)核心Servlet的一些定制化設(shè)置,如下圖05:
比如可以加進(jìn)去一些自己需要的過濾器,如下圖06:
比如可以對(duì)文件上傳進(jìn)行一些配置,如下圖07:
當(dāng)然,還可以介入到Spring容器的初始化過程中,進(jìn)行一些額外的操作,如激活特定的Profile等,如下圖08:
啟動(dòng)過程中做的事情
其實(shí)前面已經(jīng)說了一些了,這里再來個(gè)完整版的,主要包括的事情有:
1、創(chuàng)建根容器。
2、然后把根容器放入ServletContext中。
3、接著創(chuàng)建Servlet容器。
4、然后使用Servlet容器去創(chuàng)建核心Servlet。
5、接著把核心Servlet注冊(cè)到ServletContext中。
6、接著再注冊(cè)一些過濾器。
下面我們使用幾張圖把這些內(nèi)容一個(gè)個(gè)展示一下,需要詳細(xì)了解的可以去看看對(duì)應(yīng)的源碼。
創(chuàng)建根容器,可以看到是基于注解的容器類,如下圖09:
將根容器放入ServletContext中,因?yàn)镾ervletContext在應(yīng)用運(yùn)行期間一直存在,所以根容器是一個(gè)全局性的,也一直存在,如下圖10:
創(chuàng)建Servlet容器,容器類也是基于注解的,和根容器類是一樣的,如下圖11:
使用Servlet容器去創(chuàng)建核心Servlet,如下圖12:
把核心Servlet注冊(cè)到ServletContext中,如下圖13:
最后就是注冊(cè)一些過濾器了,如下圖14:
號(hào)主提示:建議做Java開發(fā)且渴望優(yōu)秀的朋友一定要把這些東西搞清楚。