springMVC之WebApplicationContext、DispatcherServlet与web容器的ServletContext

学习源码过程中,对各种context(上下文)表示很懵逼。特地留此一篇。

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

a) ServletContext: tomcat启动会创建一个ServletContext,作为全局上下文以及spring容器的宿主环境。当执行Servlet的init()方法时,会触发ServletContextListener的 public void contextInitialized(ServletContextEvent sce);方法
在这里插入图片描述

b)WebApplicationContext: 在web.xml(上图)中我们配置了ContextLoaderListener,该listener实现了ServletContextListener的contextInitialized方法用来监听Servlet初始化事件。

下图中红框部门的注释解释了该方法的作用。即初始化根上下文(即IOC容器),也就是WebApplicationContext。该类是一个接口类,其默认实现为XmlWebApplicationContext。
在这里插入图片描述
在initWebApplicationContext这个方法中进行了创建根上下文,并将该上下文以key-value的方式存储到ServletContext中
在这里插入图片描述

WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE为key,this.context则为value。this.context就是刚才创建的根上下文。后面就可以通过这个ServletContext通过这个key获取该上下文了。
而在web.xml中还有一对重要的标签
该标签内的的值是固定的原因在这张图上。该常量的值就是contextConfigLocation。通过该方法去寻找定义spring的xml文件。来初始化IOC容器的相关信息。
在这里插入图片描述

c) DispatcherServlet的上下文: 在WebApplicationContext初始化完后。开始初始化web.xml中的servlet。这个servlet可以有多个。默认我们都使用DispatcherServlet。标签中可以有标签用来配置一些DispatcherServlet的初始化参数。

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

中的initWebApplicationContext()方法中。
在这里插入图片描述

在这里插入图片描述

initWebApplicationContext()方法中的第一个红色框内就是去获取之前存在Servlet中的WebApplicationContext。通过上面说的WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE作为key
取到之后,设置为当前DispatcherServlet的父上下文。并且也把该上下文存在ServletContext中。方法如下:
在这里插入图片描述

a). 通过以上的流程,可以做到各个上下文之间既可以拥有自己独立的Bean,也可以访问各个Servlet相同的Bean

b). 通过init方法创建的dispatcherServlet上下文可以访问通过ServletContextListener中创建的WebApplicationContext上下文中的bean,反之则不行。因为WebApplicationContext是dispatcherServlet上下文

总结如下:
1.web容器可以说就是Servlet容器:ServletContext,启动tomcat必然有这个。web.xml中配置有具体的Servlet,这些都会放在Servlet容器中。

2.dispatcherServlet只是一个具体的Servlet,必然装在Servlet容器里。当然Servlet容器可以装其它任何Servlet,不一定必须有dispatcherServlet。

3.WebApplicationContext是IOC容器,里面是装spring的依赖注入产生的各种bean组件的。可以说与上面的Servlet容器及具体的Servlet没有任何直接联系。通常的IOC容器本来就不仅是给web应用用的。

4.但是web应用中具体的Servlet处理业务中一般都会使用IOC容器里的组件,所以两个容器之间要有直接引用关系。但是两个容器里的具体内容不应该有直接的引用关系。如同你跨部门协调的时候,总要找他们的领导,不会直接找具体的人员。

5.所以WebApplicationContext容器会放在ServletContext容器中,这个过程是web.xml中的监听器,监听Servlet容器的启动过程中,先产生IOC容器,并放置在ServletContext里面的。

6.如果是做其它java应用,可以使用其它IOC容器,比如
ApplicationContext (ClasspathXmlApplicationContext,FileSystemXmlApplicationContext),
可以在你启动自己应用的时候初始化这个IOC容器对象,并放在一个全局的地方引用着。其它代码就可以从IOC里取自己要的组件了。

猜你喜欢

转载自blog.csdn.net/weixin_42868638/article/details/82960991