spring启动流程
ServletContext
Web容器在启动的过程中,会为每个Web应用程序创建一个对应的ServletContext对象,它代表了当前的Web应用,为Spring IoC容器提供宿主环境。
在部署Web工程的时候,Web容器会读取web.xml,创建ServletContext,当前Web工程所有部分都共享这个Context。context-param为ServletContext提供键值对,即Servlet上下文的信息,这些信息Listener、Filter和Servlet都有可能使用到,因此先加载context-param,创建ServletContext,然后加载Listener,再加载Filter,最后加载Servlet。
接下来我将按照这个加载顺序来分析Spring容器的启动过程。
ContextLoaderListener
web.xml中配置有ContextLoaderListener,也可以自定义一个实现了ServletContextListener接口的Listener类,web.xml中的配置实例如下。
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
Web容器在启动的过程中会触发ServletContextEvent事件,会被ContextLoaderListener监听到,并调用ContextLoaderListener中的contextInitialized方法,contextInitialized方法如下所示。
public void contextInitialized(ServletContextEvent event) {
this.initWebApplicationContext(event.getServletContext());
}
ContextLoaderListener类继承了ContextLoader,在初始化Context的过程中,调用ContextLoader的initWebApplicationContext方法初始化WebApplicationContext。WebApplicationContext是一个接口,Spring默认的实现类为XmlWebApplicationContext,XmlWebApplicationContext就是Spring的IoC容器。
在初始化XmlWebApplicationContext之前,Web容器已经加载了context-param,web.xml中的context-param实例如下所示。作为Spring的IoC容器,其对应的Bean定义的配置正是context-param指定的。
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:applicationContext.xml</param-value>
</context-param>
接着进入到initWebApplicationContext方法内,initWebApplicationContext方法定义如下(已省略部分代码)。
public WebApplicationContext initWebApplicationContext(ServletContext servletContext) {
if(servletContext.getAttribute(
WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE) != null) {
throw new IllegalStateException("...");
}else{
if(this.context == null) {
this.context = this.createWebApplicationContext(servletContext);
}
}
servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, this.context);
}
在Spring IoC容器初始化前,initWebApplicationContext先检测以ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE为key的值是否为空,若不为空,则初始化IoC Context,并在初始化完毕后,以ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE为key将IoC Context存储到ServletContext中。
初始化Servlet
Servlet可以在web.xml中配置多个,在Spring中,最基本的Servlet为DispatcherServlet,对应的配置实例如下所示。
<servlet>
<servlet-name>appServlet</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>/WEB-INF/spring/appServlet/appServlet-context.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
DispatcherServlet会建立自己的IoC Context,用以持有相关的Bean,在初始化自己的IoC Context的过程中,先通过WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE,从ServletContext中获取WebApplicationContext,将WebApplicationContext作为DispatcherServlet的IoC Context的 parent Context。DispatcherServlet自己的IoC Context的初始化工作在DispatcherServlet的initStrategies方法中完成,包括控制器映射,视图解析等,initStrategies方法如下所示。
protected void initStrategies(ApplicationContext context) {
this.initMultipartResolver(context);
this.initLocaleResolver(context);
this.initThemeResolver(context);
this.initHandlerMappings(context);
this.initHandlerAdapters(context);
this.initHandlerExceptionResolvers(context);
this.initRequestToViewNameTranslator(context);
this.initViewResolvers(context);
this.initFlashMapManager(context);
}
DispatcherServlet自己的IoC Context的类型也是XmlWebApplicationContext,初始化完毕后,Spring将以与DispatcherServlet的servlet-name属性相关的符号作为key,将IoC Context保存到 ServletContext中。这样每个Servlet就都可以持有自己的Context,也就是都拥有自己的Bean空间,同时,各个Servlet之间还共享着key为WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE的WebApplicationContext,其中定义的Bean为各个Servlet共享的Bean。
//----------------------
-
首先,对于一个web应用,其部署在web容器中,web容器提供其一个全局的上下文环境,这个上下文就是ServletContext,其为后面的spring IoC容器提供宿主环境;
-
其次,在web.xml中会提供有contextLoaderListener。在web容器启动时,会触发容器初始化事件,此时contextLoaderListener会监听到这个事件,其contextInitialized方法会被调用,在这个方法中,spring会初始化一个启动上下文,这个上下文被称为根上下文,即WebApplicationContext,这是一个接口类,确切的说,其实际的实现类是XmlWebApplicationContext。这个就是spring的IoC容器,其对应的Bean定义的配置由web.xml中的context-param标签指定。在这个IoC容器初始化完毕后,spring以WebApplicationContext.ROOTWEBAPPLICATIONCONTEXTATTRIBUTE为属性Key,将其存储到ServletContext中,便于获取;
-
再次,contextLoaderListener监听器初始化完毕后,开始初始化web.xml中配置的Servlet,这个servlet可以配置多个,以最常见的DispatcherServlet为例,这个servlet实际上是一个标准的前端控制器,用以转发、匹配、处理每个servlet请求。DispatcherServlet上下文在初始化的时候会建立自己的IoC上下文,用以持有spring mvc相关的bean。在建立DispatcherServlet自己的IoC上下文时,会利用WebApplicationContext.ROOTWEBAPPLICATIONCONTEXTATTRIBUTE先从ServletContext中获取之前的根上下文(即WebApplicationContext)作为自己上下文的parent上下文。有了这个parent上下文之后,再初始化自己持有的上下文。这个DispatcherServlet初始化自己上下文的工作在其initStrategies方法中可以看到,大概的工作就是初始化处理器映射、视图解析等。这个servlet自己持有的上下文默认实现类也是mlWebApplicationContext。初始化完毕后,spring以与servlet的名字相关(此处不是简单的以servlet名为Key,而是通过一些转换,具体可自行查看源码)的属性为属性Key,也将其存到ServletContext中,以便后续使用。这样每个servlet就持有自己的上下文,即拥有自己独立的bean空间,同时各个servlet共享相同的bean,即根上下文(第2步中初始化的上下文)定义的那些bean。
//------------------------------------
作者:yang2yang
链接:https://www.jianshu.com/p/5226c7fec43a
來源:简书