# Spring中WebApplicationContext、DispatcherServlet与web容器的ServletContext关系梳理

要了解各个上下文之间的关系。首先走一遍spring在web容器(tomcat)中的启动过程

* **ServletContext:  tomcat启动会创建一个ServletContext，作为全局上下文以及spring容器的宿主环境。当执行Servlet的init()方法时，会触发ServletContextListener的 public void contextInitialized(ServletContextEvent sce);方法**

![](https://2496176152-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-M2QYpCfhA_mjmZElHWT%2F-M34IFEldCE5MYDmSxTq%2F-M34IKkel04Z_WciMtUf%2Fimport-web-01.png?generation=1584931901710964\&alt=media)

* **WebApplicationContext:  在web.xml(上图)中我们配置了ContextLoaderListener，该listener实现了ServletContextListener的contextInitialized方法用来监听Servlet初始化事件。下图中红框部门的注释解释了该方法的作用。即初始化根上下文（即IOC容器），也就是WebApplicationContext。该类是一个接口类，其默认实现为XmlWebApplicationContext。**

![](https://2496176152-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-M2QYpCfhA_mjmZElHWT%2F-M34IFEldCE5MYDmSxTq%2F-M34IKkg1EFK0pGf_yOm%2Fimport-web-02.png?generation=1584931901825030\&alt=media)在initWebApplicationContext这个方法中进行了创建根上下文，并将该上下文以key-value的方式存储到ServletContext中

![](https://2496176152-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-M2QYpCfhA_mjmZElHWT%2F-M34IFEldCE5MYDmSxTq%2F-M34IKki42tky8YG5GYM%2Fimport-web-03.png?generation=1584931902037555\&alt=media)以WebApplicationContext.ROOT\_WEB\_APPLICATION\_CONTEXT\_ATTRIBUTE为key，this.context则为value。this.context就是刚才创建的根上下文。后面就可以通过这个ServletContext通过这个key获取该上下文了。而在web.xml中还有一对重要的标签

\<context-param>该标签内的\<param-name>的值是固定的原因在这张图上。该常量的值就是contextConfigLocation。通过该方法去寻找定义spring的xml文件。来初始化IOC容器的相关信息。

![](https://2496176152-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-M2QYpCfhA_mjmZElHWT%2F-M34IFEldCE5MYDmSxTq%2F-M34IKkku5k5xdN1BMUB%2Fimport-web-04.png?generation=1584931902199440\&alt=media)

* **DispatcherServlet的上下文:  在WebApplicationContext初始化完后。开始初始化web.xml中的servlet。这个servlet可以有多个。默认我们都使用DispatcherServlet。\<servlet>标签中可以有\<init-param>标签用来配置一些DispatcherServlet的初始化参数。可以参考**[**核心流程剖析及原理分析**](https://tuonioooo-notebook.gitbook.io/application-framework/springyuan-ma-jie-du-pian/springmvcyuan-ma-jie-du/he-xin-liu-cheng-pou-xi)

该servlet初始化流程是有tomcat的Servlet的init()方法触发。DispatcherServleet-继承->FrameworkServlet-继承->HttpServletBean-继承-GenericServlet- 实现 ->Servlet。这样的一条关系链。其核心方法在FrameworkServlet中的initServletBean()中的initWebApplicationContext()方法中。

![](https://2496176152-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-M2QYpCfhA_mjmZElHWT%2F-M34IFEldCE5MYDmSxTq%2F-M34IKkmw7-f5dHxoKbe%2Fimport-web-05.png?generation=1584931901112805\&alt=media)

![](https://2496176152-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-M2QYpCfhA_mjmZElHWT%2F-M34IFEldCE5MYDmSxTq%2F-M34IKkoaOeRGe0yrF-o%2Fimport-web-06.png?generation=1584931901409162\&alt=media)initWebApplicationContext()方法中的第一个红色框内就是去获取之前存在Servlet中的WebApplicationContext。通过上面说的WebApplicationContext.ROOT\_WEB\_APPLICATION\_CONTEXT\_ATTRIBUTE作为key取到之后，设置为当前DispatcherServlet的父上下文。并且也把该上下文存在ServletContext中。方法如下：

## ![](https://2496176152-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-M2QYpCfhA_mjmZElHWT%2F-M34IFEldCE5MYDmSxTq%2F-M34IKkqe_kVRvYB-_lC%2Fimport-web-07.png?generation=1584931901424220\&alt=media)

## 总结

* 通过以上的流程，可以做到各个上下文之间既可以拥有自己独立的Bean，也可以访问各个Servlet相同的Bean
* 通过init方法创建的dispatcherServlet上下文可以访问通过ServletContextListener中创建的WebApplicationContext上下文中的bean，反之则不行。因为WebApplicationContext是dispatcherServlet上下文的父容器。
