Spring容器啟動源碼解析
- 2019 年 10 月 25 日
- 筆記
1. 前言
最近搭建的工程都是基於SpringBoot,簡化配置的感覺真爽。但有個以前的項目還是用SpringMvc寫的,看到滿滿的配置xml文件,卻有一種想去深入了解的衝動。折騰了好幾天,決心去寫這篇關於Spring啟動的博客,自己是個剛入職的小白,技術水平有限,也是硬着頭皮看源碼去Debug,很多不懂的地方還請諒解!
2. 概述
先給出幾個讓我頭皮發麻的概念:web容器,Spring容器,SpringMvc容器
容器就是管理對象的地方,例如web容器就是管理servlet的地方,Spring容器就是管理Service,dao等Bean的地方,SpringMvc就是管理Controller等bean的地方(下文會做解釋)。一個SpringMvc項目的啟動離不開上述三個容器。所以這就是這篇文章的講點,各個容器的啟動過程解析。
3. Web容器初始化過程
官方文檔是對於Web容器初始化時是這樣描述的(英文不懂,已翻譯成中文)
1. 部署描述文件(web.xml)中的<listener>標記的監聽器會被創建和初始化
2. 對於實現了ServletContextListener的監聽器,會執行它的初始化方法 contextInitialized()
3. 部署描述文件中的<filter>標記的過濾器會被創建和初始化,調用其init()方法
4. 部署描述文件中的<servlet>標記的servlet會根據<load-on-startup>中的序號創建和初始化,調用init()方法
大致流程了解之後,結合自己的SpringMvc項目一步步深入,先貼一下基本的web.xml文件
<?xml version="1.0" encoding="UTF-8"?> <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:web="http://java.sun.com/xml/ns/javaee" xmlns="http://java.sun.com/xml/ns/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" id="WebApp_ID" version="2.5"> <display-name>dmpserver</display-name> <welcome-file-list> <welcome-file>login.jsp</welcome-file> </welcome-file-list> <context-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:spring.xml</param-value> </context-param> <context-param> <param-name>log4jConfigLocation</param-name> <param-value>classpath:log4jConfig.xml</param-value> </context-param> <filter> <filter-name>encodingFilter</filter-name> <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class> <init-param> <param-name>encoding</param-name> <param-value>utf-8</param-value> </init-param> </filter> <filter-mapping> <filter-name>encodingFilter</filter-name> <url-pattern>/*</url-pattern> </filter-mapping> <listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener> <listener> <listener-class>org.springframework.web.util.Log4jConfigListener</listener-class> </listener> <servlet> <description>spring mvc servlet</description> <servlet-name>rest</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <init-param> <param-name>contextConfigLocation</param-name> <param-value> classpath:spring-mvc.xml </param-value> </init-param> <load-on-startup>1</load-on-startup> </servlet> </web-app>
1. 容器會先解析<context-param>中的鍵值對(上述代碼重點關注Spring配置文件Spring.xml)
2. 容器創建一個application內置對象servletContext(可以理解為servlet上下文或web容器),用於全局變量共享
3. 將解析的<context-param>鍵值對存放在application即servletContext中
4. 讀取<listener>中的監聽器,一般會使用ContextLoaderListener類,調用其contextInitialized方法,創建IOC容器(Spring容器)webApplicationContext。將webApplication容器放入application(servlet上下文)中作為根IOC容器,鍵名為WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE 注意的是,webApplicationContext是全局唯一的,一個web應用只能有一個根IOC容器。因為這個根IOC容器是讀取<context-param>配置的鍵值對來創建Bean,這個根IOC容器只能訪問spring.xml中配置的Bean,我們在Spring.xml中一般配置的是service,dao等Bean。所以根IOC容器(Spring容器)只能管理service,dao等Bean
5. listener加載完畢後,加載filter過濾器
6. 加載servlet,一般springMvc項目中會優先加載 DispatcherServlet(現在開始加載SpringMvc容器了)
7. DispatcherServlet的父類FrameworkServlet重寫了其父類的initServletBean()方法,在初始化時調用initWebApplicationContext()方法和onRefresh()方法
8. initWebApplicationContext()方法會在servletContext(即當前servlet上下文)創建一個子IOC容器(即SpringMvc容器),如果存在上述的根IOC容器,就設置根IOC容器作為父容器,如果不存在,就將父容器設置為NULL
9. 讀取<servlet>標籤的<init-param>配置的xml文件並加載相關Bean。此時加載的是Spring-mvc.xml配置文件,管理的是Controller等Bean
10. onRefresh()加載其他組件
4. 啟動過程分析
4.1 listener初始化Spring容器
tomcat啟動後,<context-param>標籤的內容讀取後會被放進application中,做為Web應用的全局變量使用,接下來創建listener時會使用到這個全局變量,因此,Web應用在容器中部署後,進行初始化時會先讀取這個全局變量,之後再進行上述講解的初始化啟動過程。
查看ContextLoaderListener源碼
public class ContextLoaderListener extends ContextLoader implements ServletContextListener { public ContextLoaderListener() { } public ContextLoaderListener(WebApplicationContext context) { super(context); } public void contextInitialized(ServletContextEvent event) { this.initWebApplicationContext(event.getServletContext()); } public void contextDestroyed(ServletContextEvent event) { this.closeWebApplicationContext(event.getServletContext()); ContextCleanupListener.cleanupAttributes(event.getServletContext()); } }
據官方文檔說明,實現ServletContextListener接口,執行contextInitialized(),進入initWebApplicationContext方法。contextInitialized()和contextDestroyed()方法會在web容器啟動或銷毀時執行。網上查了下此處設計模式用到的是觀察者模式和代理模式,自己也不懂就不做詳解了
查看ContextLoader.class中的initWebApplicationContext方法
public WebApplicationContext initWebApplicationContext(ServletContext servletContext) { /* 首先通過WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE 這個String類型的靜態變量獲取一個根IoC容器,根IoC容器作為全局變量 存儲在application對象中,如果存在則有且只能有一個 如果在初始化根WebApplicationContext即根IoC容器時發現已經存在 則直接拋出異常,因此web.xml中只允許存在一個ContextLoader類或其子類的對象 */ if (servletContext.getAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE) != null) { throw new IllegalStateException("Cannot initialize context because there is already a root application context present - check whether you have multiple ContextLoader* definitions in your web.xml!"); } else { Log logger = LogFactory.getLog(ContextLoader.class); servletContext.log("Initializing Spring root WebApplicationContext"); if (logger.isInfoEnabled()) { logger.info("Root WebApplicationContext: initialization started"); } long startTime = System.currentTimeMillis(); try { if (this.context == null) { // 創建一個根IOC容器 this.context = this.createWebApplicationContext(servletContext); } if (this.context instanceof ConfigurableWebApplicationContext) { ConfigurableWebApplicationContext cwac = (ConfigurableWebApplicationContext)this.context; if (!cwac.isActive()) { if (cwac.getParent() == null) { // 為根IOC容器設置一個父容器 ApplicationContext parent = this.loadParentContext(servletContext); cwac.setParent(parent); } this.configureAndRefreshWebApplicationContext(cwac, servletContext); } } //將創建好的IoC容器放入到application對象中,並設置key為WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, this.context); ClassLoader ccl = Thread.currentThread().getContextClassLoader(); if (ccl == ContextLoader.class.getClassLoader()) { currentContext = this.context; } else if (ccl != null) { currentContextPerThread.put(ccl, this.context); } if (logger.isDebugEnabled()) { logger.debug("Published root WebApplicationContext as ServletContext attribute with name [" + WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE + "]"); } if (logger.isInfoEnabled()) { long elapsedTime = System.currentTimeMillis() - startTime; logger.info("Root WebApplicationContext: initialization completed in " + elapsedTime + " ms"); } return this.context; } catch (RuntimeException var8) { logger.error("Context initialization failed", var8); servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, var8); throw var8; } catch (Error var9) { logger.error("Context initialization failed", var9); servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, var9); throw var9; } } }
initWebApplicationContext方法的主要目的是創建一個根IOC容器,並放入servlet上下文中。看上述源碼可知,根IOC容器只能僅有一個,作為全局變量存儲在servletContext中。將根IoC容器放入到application對象之前進行了IoC容器的配置和刷新操作,調用了configureAndRefreshWebApplicationContext()方法,該方法源碼如下:
protected void configureAndRefreshWebApplicationContext(ConfigurableWebApplicationContext wac, ServletContext sc) { String configLocationParam; if (ObjectUtils.identityToString(wac).equals(wac.getId())) { configLocationParam = sc.getInitParameter("contextId"); if (configLocationParam != null) { wac.setId(configLocationParam); } else { wac.setId(ConfigurableWebApplicationContext.APPLICATION_CONTEXT_ID_PREFIX + ObjectUtils.getDisplayString(sc.getContextPath())); } } wac.setServletContext(sc); /* 在容器啟動時,會把<context-param>中的內容放入servlet上下文的全局變量中, 此時獲取key為contextConfigLocation的變量,及Spring.xml配置文件 將其放入到webApplicationContext中 */ configLocationParam = sc.getInitParameter("contextConfigLocation"); if (configLocationParam != null) { wac.setConfigLocation(configLocationParam); } ConfigurableEnvironment env = wac.getEnvironment(); if (env instanceof ConfigurableWebEnvironment) { ((ConfigurableWebEnvironment)env).initPropertySources(sc, (ServletConfig)null); } this.customizeContext(sc, wac); wac.refresh(); }
configureAndRefreshWebApplicationContext方法比較重要的是把配置文件信息放入根IOC容器中。方法最後調用了refresh()方法,對配置文件信息(Bean)進行加載。因為refresh()這是個ConfigurableApplication-Context接口方法,想到了它的常用實現類ClassPathXmlApplicationContext,一層層進去找到了Abstract-ApplicationContext,實現了refresh(),見如下源碼:
public void refresh() throws BeansException, IllegalStateException { Object var1 = this.startupShutdownMonitor; synchronized(this.startupShutdownMonitor) { this.prepareRefresh(); ConfigurableListableBeanFactory beanFactory = this.obtainFreshBeanFactory(); this.prepareBeanFactory(beanFactory); try { this.postProcessBeanFactory(beanFactory); this.invokeBeanFactoryPostProcessors(beanFactory); this.registerBeanPostProcessors(beanFactory); this.initMessageSource(); this.initApplicationEventMulticaster(); this.onRefresh(); this.registerListeners(); this.finishBeanFactoryInitialization(beanFactory); this.finishRefresh(); } catch (BeansException var9) { if (this.logger.isWarnEnabled()) { this.logger.warn("Exception encountered during context initialization - cancelling refresh attempt: " + var9); } this.destroyBeans(); this.cancelRefresh(var9); throw var9; } finally { this.resetCommonCaches(); } } }
該方法主要用於創建並初始化contextConfigLocation類配置的xml文件中的Bean,因此,如果我們在配置Bean時出錯,在Web應用啟動時就會拋出異常,而不是等到運行時才拋出異常。因為技術能力有限加上此處方法太多,就不在一一解析了。到此為止,整個Spring容器加載完畢,下面開始加載SpringMVC容器
4.2 Filter初始化
因為Filter的操作沒有涉及IOC容器,此處不做詳解,上面web.xml中配置的是一個UTF8編碼過濾器
5. 總結
時間有限,只大致介紹了Spring容器的初始化,後面還沒來得及整理,對於springMvc容器的創建和初始化下篇文章見