SpringMVC中的DispatcherServlet初始化流程詳解
DispatcherServlet初始化流程
分析一下DispatcherServlet這個(gè)前端控制器的啟動和初始化的整個(gè)過程
通過前面分析已經(jīng)知道了DispatcherServlet這個(gè)前端控制器是一個(gè)Servlet了,所以生命周期和普通的Servlet是差不多的,在一個(gè)Servlet初始化的時(shí)候都會調(diào)用該Servlet的init()方法。下面這個(gè)是DispatcherSerlvet父類HttpServletBean中的init方法。

我們發(fā)現(xiàn)這里會調(diào)用initServletBean()方法進(jìn)行具體的初始化,而該類這個(gè)方法的具體實(shí)現(xiàn)這是其子類FrameworkServlet。主要邏輯就是初始化上下文。
protected final void initServletBean() throws ServletException {
this.getServletContext().log("Initializing Spring " + this.getClass().getSimpleName() + " \'" + this.getServletName() + "\'");
if(this.logger.isInfoEnabled()) {
this.logger.info("Initializing Servlet \'" + this.getServletName() + "\'");
}
long startTime = System.currentTimeMillis();
try {
/**
這里初始化上下文
**/
this.webApplicationContext = this.initWebApplicationContext();
this.initFrameworkServlet();
} catch (RuntimeException | ServletException var4) {
this.logger.error("Context initialization failed", var4);
throw var4;
}
if(this.logger.isDebugEnabled()) {
String value = this.enableLoggingRequestDetails?"shown which may lead to unsafe logging of potentially sensitive data":"masked to prevent unsafe logging of potentially sensitive data";
this.logger.debug("enableLoggingRequestDetails=\'" + this.enableLoggingRequestDetails + "\': request parameters and headers will be " + value);
}
if(this.logger.isInfoEnabled()) {
this.logger.info("Completed initialization in " + (System.currentTimeMillis() - startTime) + " ms");
}
}繼續(xù)跟進(jìn)initWebApplicationContext這個(gè)方法
protected WebApplicationContext initWebApplicationContext() {
/**
若webApplicationContext不為空的時(shí)候從SerlvetContext去出根上下文作為它的雙親上下文
**/
WebApplicationContext rootContext = WebApplicationContextUtils.getWebApplicationContext(this.getServletContext());
WebApplicationContext wac = null;
if(this.webApplicationContext != null) {
wac = this.webApplicationContext;
if(wac instanceof ConfigurableWebApplicationContext) {
ConfigurableWebApplicationContext attrName = (ConfigurableWebApplicationContext)wac;
if(!attrName.isActive()) {
if(attrName.getParent() == null) {
attrName.setParent(rootContext);
}
this.configureAndRefreshWebApplicationContext(attrName);
}
}
}
if(wac == null) {
wac = this.findWebApplicationContext();
}
if(wac == null) {
wac = this.createWebApplicationContext(rootContext);
}
//這里通過調(diào)用子類DispatcherServlet實(shí)現(xiàn)的onRefresh方法
if(!this.refreshEventReceived) {
this.onRefresh(wac);
}
//把當(dāng)前建立好的上下文存到ServletContext里面去
if(this.publishContext) {
String attrName1 = this.getServletContextAttributeName();
this.getServletContext().setAttribute(attrName1, wac);
}
return wac;
}繼續(xù)跟進(jìn)這個(gè)onRefresh方法,發(fā)現(xiàn)是一個(gè)模板方法,其具體實(shí)現(xiàn)子類則是DispatcherServlet。

跳轉(zhuǎn)到DispatcherServlet中查看該方法,發(fā)現(xiàn)里面又調(diào)用了initStrategies這個(gè)方法

這時(shí)我們就大致了解了,這個(gè)DispatcherServlet初始化的過程了,首先DispatcherServlet持有者一個(gè)以自己的Servlet名字命名的Ioc容器,也就是我們看到的WebApplicationContext對象,這個(gè)Ioc容器建立起來后,與Web容器相關(guān)的各種配置加載也都完成了。并且這個(gè)初始化的入口就是由最初的HttpServletBean的init方法觸發(fā)的,因?yàn)檫@個(gè)HttpServletBean是HttpServlet的子類,接下來HttpServletBean的子類FrameworkServlet對Ioc容器進(jìn)行了初始化操作,并且利用onRefresh方法回調(diào)了DispatcherServlet中的initStrategies方法,在這個(gè)方法里啟動了整個(gè)SpringMVC框架了,我們繼續(xù)往下面跟進(jìn)看看。
//該屬性默認(rèn)為true
private boolean detectAllHandlerMappings = true;
private void initHandlerMappings(ApplicationContext context) {
this.handlerMappings = null;
//這里面的邏輯是從導(dǎo)入所有的HandlerMappingBean,這些Bean有可能存在與雙親容器中,也可能在DispathcerServlet持有的容器的,這里detectAllHandlerMappings默認(rèn)為true,默認(rèn)從所有容器中導(dǎo)入
if(this.detectAllHandlerMappings) {
Map hm = BeanFactoryUtils.beansOfTypeIncludingAncestors(context, HandlerMapping.class, true, false);
if(!hm.isEmpty()) {
this.handlerMappings = new ArrayList(hm.values());
AnnotationAwareOrderComparator.sort(this.handlerMappings);
}
} else {
//否則通過直接通過名字從當(dāng)前的IOC容器中通過getBean方法獲取handlerMapping
try {
HandlerMapping hm1 = (HandlerMapping)context.getBean("handlerMapping", HandlerMapping.class);
this.handlerMappings = Collections.singletonList(hm1);
} catch (NoSuchBeanDefinitionException var3) {
;
}
}
//如果還是沒有找到hadlerMapping就需要設(shè)定默認(rèn)的handlerMappings了
if(this.handlerMappings == null) {
this.handlerMappings = this.getDefaultStrategies(context, HandlerMapping.class);
if(this.logger.isTraceEnabled()) {
this.logger.trace("No HandlerMappings declared for servlet \'" + this.getServletName() + "\': using default strategies from DispatcherServlet.properties");
}
}
}下面是用debugger看看它究竟獲取了那些handlerMapping,如下:7個(gè)handlerMapping

除了這個(gè)初始化handlerMapping的initHandlerMapping方法,當(dāng)然還初始化了很多東西,如支持國際化的LocalResolver以及視圖生成的ViewResolver等的初始化過程,其余的有興趣自己跟著看一下,這里就不細(xì)跟了。到這里我們就知道了整個(gè)DispatcherServlet的初始化的大體流程了。
到此這篇關(guān)于SpringMVC中的DispatcherServlet初始化流程詳解的文章就介紹到這了,更多相關(guān)DispatcherServlet初始化流程內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot+EasyPoi實(shí)現(xiàn)excel導(dǎo)出功能
最新小編遇到這樣一個(gè)需求,根據(jù)檢索條件查詢列表并將結(jié)果導(dǎo)出到excel,實(shí)現(xiàn)過程也非常簡單,感興趣的朋友跟隨小編一起看看吧2021-09-09
Java中利用POI優(yōu)雅的導(dǎo)出Excel文件詳解
這篇文章主要給大家介紹了關(guān)于Java中如何利用POI優(yōu)雅的導(dǎo)出Excel文件的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用Java具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05
解決springboot 實(shí)體類String轉(zhuǎn)Date類型的坑
這篇文章主要介紹了解決springboot 實(shí)體類String轉(zhuǎn)Date類型的坑,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10
SpringBoot實(shí)現(xiàn)在webapp下直接訪問html,jsp
這篇文章主要介紹了SpringBoot實(shí)現(xiàn)在webapp下直接訪問html,jsp問題,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-10-10
CMD運(yùn)行Intellij Idea編譯后的class文件操作
這篇文章主要介紹了CMD運(yùn)行Intellij Idea編譯后的class文件操作,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-02-02
java數(shù)據(jù)結(jié)構(gòu)與算法之冒泡排序詳解
這篇文章主要介紹了java數(shù)據(jù)結(jié)構(gòu)與算法之冒泡排序,結(jié)合實(shí)例形式詳細(xì)分析了java冒泡排序的原理、實(shí)現(xiàn)技巧與相關(guān)注意事項(xiàng),需要的朋友可以參考下2017-05-05
SpringBoot Filter修改返回內(nèi)容,解決請求卡死200的錯(cuò)誤
這篇文章主要介紹了SpringBoot Filter修改返回內(nèi)容,解決請求卡死200的錯(cuò)誤問題,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-07-07

