学习源码过程当中,对各类context(上下文)表示很懵逼。特意留此一篇。html
1.要了解各个上下文之间的关系。首先走一遍spring在web容器(tomcat)中的启动过程web
a) ServletContext: tomcat启动会建立一个ServletContext,做为全局上下文以及spring容器的宿主环境。当执行Servlet的init()方法时,会触发ServletContextListener的 public void contextInitialized(ServletContextEvent sce);方法spring
b)WebApplicationContext: 在web.xml(上图)中咱们配置了ContextLoaderListener,该listener实现了ServletContextListener的contextInitialized方法用来监听Servlet初始化事件。api
下图中红框部门的注释解释了该方法的做用。即初始化根上下文(即IOC容器),也就是WebApplicationContext。该类是一个接口类,其默认实现为XmlWebApplicationContext。tomcat
在initWebApplicationContext这个方法中进行了建立根上下文,并将该上下文以key-value的方式存储到ServletContext中学习
以WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE为key,this.context则为value。this.context就是刚才建立的根上下文。后面就能够经过这个ServletContext经过这个key获取该上下文了。而在web.xml中还有一对重要的标签this
<context-param>该标签内的<param-name>的值是固定的缘由在这张图上。该常量的值就是contextConfigLocation。经过该方法去寻找定义spring的xml文件。来初始化IOC容器的相关信息。spa
c) DispatcherServlet的上下文: 在WebApplicationContext初始化完后。开始初始化web.xml中的servlet。这个servlet能够有多个。默认咱们都使用DispatcherServlet。<servlet>标签中能够有<init-param>标签用来配置一些DispatcherServlet的初始化参数。.net
该servlet初始化流程是有tomcat的Servlet的init()方法触发。DispatcherServleet-继承->FrameworkServlet-继承->HttpServletBean-继承-GenericServlet- 实现 ->Servlet。这样的一条关系链。其核心方法在FrameworkServlet中的initServletBean()中xml
中的initWebApplicationContext()方法中。
initWebApplicationContext()方法中的第一个红色框内就是去获取以前存在Servlet中的WebApplicationContext。经过上面说的WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE做为key
取到以后,设置为当前DispatcherServlet的父上下文。而且也把该上下文存在ServletContext中。方法以下
2.
a). 经过以上的流程,能够作到各个上下文之间既能够拥有本身独立的Bean,也能够访问各个Servlet相同的Bean
b). 经过init方法建立的dispatcherServlet上下文能够访问经过ServletContextListener中建立的WebApplicationContext上下文中的bean,反之则不行。由于WebApplicationContext是dispatcherServlet上下文的父容器。
3. api文档