序言
去年花里逢君别,今年花开又一年。
对JavaWeb中Context的学习,设计模式yyds。
名词解释
文中出现的所有名词解释:
Web容器:Tomcat,JBoss
Tomcat架构
Tomcat大家都比较熟悉了,Connector连接器负责外部交流,Container容器负责内部处理。
Container组件,容器,内部实现了4种子容器:
4种子容器: Engine、Host、Context、Wrapper
,这四种容器是父子关系。
data:image/s3,"s3://crabby-images/ffd6c/ffd6cedadad0fb6e03c77c2e50d02b581bd8277f" alt="image-20211208101735602"
- Engine: 最顶层容器组件,可以包含多个Host。实现类为
org.apache.catalina.core.StandardEngine
- Host: 代表一个虚拟主机,每个虚拟主机和某个域名Domain Name相匹配,可以包含多个Context。实现类为
org.apache.catalina.core.StandardHost
- Context: 一个Context对应于一个Web 应用,可以包含多个Wrapper。实现类为
org.apache.catalina.core.StandardContext
- Wrapper: 一个Wrapper对应一个Servlet。负责管理 Servlet ,包括Servlet的装载、初始化、执行以及资源回收。实现类为
org.apache.catalina.core.StandardWrapper
Context在Tomcat中代表一个Web应用的抽象表示,下文重点看Context部分。
三种Context
ServletContext
StandardContext
ApplicationContext
ServletContext
ServletContext是javax.servlet包下的接口(规范),它不属于Tomcat也不属于Spring家族。
它的本意是可以对某个Web应用的各种资源和功能进行访问。
Web容器在启动时,它会为每个Web应用程序都创建一个对应的ServletContext对象。它代表当前Web应用,并且它被所有客户端共享。
StandardContext
data:image/s3,"s3://crabby-images/78d83/78d83145d51fe2959cd1d936e3759a1bbad4e278" alt="image-20211208103824206"
Tomcat架构图中的Context其实就是个接口,Tomcat内部对Context接口的默认实现为StandardContext类。
StandardContext是一个Web应用在Tomcat内部的具体对象。
ApplicationContext
data:image/s3,"s3://crabby-images/989db/989db7f8b50939430ee99ab3e4c3aa1e8d05ba0e" alt="image-20211208104906459"
ApplicationContext其实更像是为了满足ServletContext规范而对StandardContext的一种封装。
在Tomcat内部,ApplicationContext是ServletContext接口的具体实现。
而Tomcat惯用Facade方式,因此我们一般获取的ServletContext实例是ApplicationContextFacade对象
ApplicationContextFacade对象其实就是对ApplicationContext做了一层包装。
搞搞StandardContext
web.xml信息装配到StandardContext中
如果你debug过tomcat的源码,你会发现,其实在tomcat启动的时候,会执行valve责任链那一套,来到
Tomcat 层级调用组件的start()方法,执行StandardContext.startInternal() 方法:
1 | //挑重点说 |
可以看出来,在startInternal()方法中调用fireLifecycleEvent()发布一个”configure_start” 事件。
fireLifecycleEvent方法内部将信号configure_start被包装为一个LifecycleEvent对象event,通知所有的lifecycleListeners。
LifecycleListener是一个接口,接口中的lifecycleEvent抽象方法被ContextConfig实现类进行了重写。
data:image/s3,"s3://crabby-images/4add3/4add34358c93662ad9db34b736370422f2e6966c" alt="image-20211208110604926"
下面是ContextConfig类重写的lifecycleEvent方法,可以看到我们现在满足信号条件会执行configureStart方法。
data:image/s3,"s3://crabby-images/ef1b1/ef1b157961b291fc00ef2e8181bf90c0d7c8a8c5" alt="image-20211208110745925"
1 | protected synchronized void configureStart() { |
动态调试源码看一眼,确实是StandardContext对象
data:image/s3,"s3://crabby-images/d8938/d8938f75a3648fab18ace544184eb0bfdea62430" alt="image-20211208111356903"
configureContext方法内部其实就是为我们配置filter、listener、filter、filterMap等信息。
1 | private void configureContext(WebXml webxml) { |
执行
configure_start信号目的就是让tomcat将web.xml信息加入到StandardContext中,现在已经结束了。
回到StandardContext.startInternal()方法中:
data:image/s3,"s3://crabby-images/120e8/120e8bf110e7c8e88d89238cb48944006f147028" alt="image-20211208113227724"
整合SpringMVC
1 | <web-app> |
我们发现,在springMVC的web.xml通常有这么一节配置:
1 | <!--监听器--> |
接着上一小节说,如果我们现在将SpringMVC工程打成war包,工程的web.xml就会被解析,这段listener就会被解析进来
我们进入listenerStart方法,顾名思义就是触发每个listener的方法:
data:image/s3,"s3://crabby-images/7eba0/7eba048e9703bd596bad42ace795e421cefa0163" alt="image-20211208114414880"
重点片段:
data:image/s3,"s3://crabby-images/b698a/b698a1a1c743d24545223951957d442427fcab6f" alt="image-20211208115857716"
debug看一下,确实触发ContextLoaderListener#contextInitialized方法,event包装的是ApplicationContextFacade
data:image/s3,"s3://crabby-images/e39a9/e39a9a9afb3e18787675227c1b482fe5710a9fe1" alt="image-20211208115335072"
data:image/s3,"s3://crabby-images/15b95/15b95a7bb1431f12285a465357e490be6c6f182b" alt="image-20211208115446263"
我们直接进入initWebApplicationContext方法
上来就是域对象属性判断:
data:image/s3,"s3://crabby-images/8ad4e/8ad4e8d1d76535bead28ef4918743597aeb7c74f" alt="image-20211208120244813"
我们传进来的ApplicationContextFacade当然没有这个属性了
来到this.context = this.createWebApplicationContext(servletContext);
很明显,这是创建SpringMVC自己的WebApplicationContext
data:image/s3,"s3://crabby-images/f4cf3/f4cf37aec6ea38b3f74c13cd28ca0bc68a916851" alt="image-20211208120622462"
搞搞ApplicationContext
Spring有两个核心接口:BeanFactory和ApplicationContext。都可以代表Spring容器。
什么是Spring容器?Spring容器是生成Bean实例的工厂,并且管理容器中的Bean的生命周期。
应用中的所有组件,都处于Spring的管理下,都被Spring以Bean的方式管理,Spring负责创建Bean实例,并管理他们的生命周期。
和 BeanFactory 类似,它可以加载配置文件中定义的 bean,将所有的 bean 集中在一起,当有请求的时候分配 bean。
另外,它增加了企业所需要的功能,比如,从属性文件从解析文本信息和将事件传递给所指定的listeners。
BeanFactory的实现是按需创建,即第一次获取Bean时才创建这个Bean,而ApplicationContext会一次性创建所有的Bean。
最常被使用的 ApplicationContext 接口实现:
- FileSystemXmlApplicationContext:该容器从 XML 文件中加载已被定义的 bean。需要提供XML 文件的完整路径。
- ClassPathXmlApplicationContext:该容器从 XML 文件中加载已被定义的 bean。不需要提供 XML 文件的完整路径。
- WebXmlApplicationContext:该容器会在一个 web 应用程序的范围内加载在 XML 文件中已被定义的 bean。
转换
我们回到”创建SpringMVC自己的WebApplicationContext”这里:
data:image/s3,"s3://crabby-images/f4cf3/f4cf37aec6ea38b3f74c13cd28ca0bc68a916851" alt="image-20211208120622462"
跟进去createWebApplicationContext方法中:
data:image/s3,"s3://crabby-images/086fc/086fc2c7394f85f63e6b5f6a6b8a16bea0a28150" alt="image-20211208132051735"
determineContextClass方法:
data:image/s3,"s3://crabby-images/64ea9/64ea936cb03ac8296fac47b05fcb4d84e7441669" alt="image-20211208132431548"
这里的ApplicationContextFacade没有contextClass属性(自己在web.xml可以配置)
那么ClassUtils.forName就为应用返回一个默认的XmlWebApplicationContext类型的Class对象。
返回上级createWebApplicationContext方法,返回值contextClass其实就是XmlWebApplicationContext类对象。
data:image/s3,"s3://crabby-images/f3d81/f3d81a62e7fe660859b8349c5376888c7c5a88be" alt="image-20211208132634934"
进入BeanUtils.instantiateClass方法,将Class对象进行了实例化,返回了XmlWebApplicationContext的实例化对象,赋值给了ContextLoader类(ContextLoaderListener的父类)的context属性。
XmlWebApplicationContext是WebApplicationContext的某一个实现类
我们可以看到,其实这里的context已经被标识为”Root WebApplicationContext”了
一路跟下来,我们知道context其实是tomcat内部的ApplicationContextFacade对象被Spring包装为自己的XmlWebApplicationContext对象了。
并且作为“Root WebApplicationContext”,是唯一的。
data:image/s3,"s3://crabby-images/7b7ed/7b7edc9f1848ed9ffcd739fc60e4246764cc843d" alt="image-20211208133818738"
ok继续向下看,parent为null,进入configureAndRefreshWebApplicationContext方法:
之后挑重点说:
1 | configureAndRefreshWebApplicationContext() |
contextConfigLocation是要告诉ContextLoaderListener要把哪些Bean注入到XmlWebApplicationContext管理的BeanFactory。
data:image/s3,"s3://crabby-images/7a6f3/7a6f34a408a6e03ce95d70ff53a801a7ca96dd10" alt="image-20211208135135541"
refresh才开始真正初始化组件。
总结:
我们在SpringMVC工程中配置好的web.xml中的listener类型ContextLoaderListener,会监听Web容器(tomcat)的初始化事件。ContextLoaderListener的contextInitialized方法中,Spring会将tomcat中唯一的ApplicationContextFacade对象包装为XMLWebApplicationContext对象并标识为全局唯一的Spring根容器”ROOT WebApplicationContext”。
融合
当我们完成bean初始化之后,还是要回到最开始的initWebApplicationContext方法
data:image/s3,"s3://crabby-images/44ca8/44ca8930703c6c67e3d380d1fafeeab01c8fc662" alt="image-20211208142522728"
是的你没有看错,二者凭借成员属性“融合”到一起了。
Spring的Web应用上下文和Web容器的上下文应用就可以实现互访,二者实现了“融合”。我中有你,你中有我!
data:image/s3,"s3://crabby-images/10152/10152951207be5411b921858e75891693ecdfb3d" alt="image-20211208143331833"
总而言之,Servlet规范中ServletContext实现(ApplicationContext)是tomcat的Context实现(StandardContext)的一个成员变量,而Spring的ApplicationContext是Servlet规范中ServletContext的一个属性。
DispatcherServlet
data:image/s3,"s3://crabby-images/e8845/e884550f53aedb32452559a6cb5467c3649cbd86" alt="image-20211208151715828"
DispatcherServlet的本质是Servlet,是前端控制器。
data:image/s3,"s3://crabby-images/b46a1/b46a13438eeb138034d784c05d4d564e843e2707" alt="image-20211208151913908"
但是既然本质是Servlet,初始化还是要走init方法的。
还记得tomcat中的StandardContext.startInternal方法:
data:image/s3,"s3://crabby-images/af9e5/af9e51d24fa60d2e7d84d6bcce9f00258ffa73b6" alt="image-20211208152332597"
进入loadOnStartup方法:
data:image/s3,"s3://crabby-images/89047/890470d79f940411b92edf27364acf9333ec3cb7" alt="image-20211208152422016"
注释已经写的很清楚了,加载那些loadonstartup=1的Servlet
现在tomcat准备加载DispatcherServlet对象,要去寻找init方法
1 | GenericServlet (javax.servlet) |
在HttpServletBean找到了init方法
data:image/s3,"s3://crabby-images/13534/1353400b7f4149800a19fff5bde3c486e306dda8" alt="image-20211208165433964"
发现调用的是FrameworkServlet类的initServletBean方法:
data:image/s3,"s3://crabby-images/76a70/76a70ceb4c11a27a3c0bfbd349136f3c707650ed" alt="image-20211208165719017"
进入initWebApplicationContext方法:
data:image/s3,"s3://crabby-images/9322f/9322f652b430199a0902a0a78e605e461b946f85" alt="image-20211208170748959"
这里创建的是特定Servlet拥有的子IoC容器
根IoC容器
做为全局共享的IoC容器
放入Web应用需要共享的Bean
,而子IoC容器
根据需求的不同,放入不同的Bean
,这样能够做到隔离,保证系统的安全性。
返回子容器
data:image/s3,"s3://crabby-images/b19be/b19be2031b65726fc5024a15d31e748aa5ef58d9" alt="image-20211208173238331"
当IoC子容器构造完成后调用了onRefresh()方法,具体实现由子类覆盖,调用onRefresh()方法时将前文创建的IoC子容器作为参数传入,DispatcherServletBean类的onRefresh()初始化配置一堆组件
data:image/s3,"s3://crabby-images/e1d75/e1d75f858a5382adfb4e1820870073c93f5f9ec1" alt="image-20211208173554027"
SpringMVC 启动流程
- tomcat会读取项目web.xml中的
<context-param>
内容、<listener>
标签、filter
标签。 - tomcat会读取项目的web.xml中的
<listener>
,如果配置了listener为ContextLoaderListener,那么就创建父容器ROOT,与tomcat的ApplicationContextFacade“融合”。 - 读取
servlet
标签,一般是DispatchServlet - 为DispatchServlet创建子容器
- 读取
<servlet>
标签的<init-param>
配置的xml文件
并加载相关Bean
- onfresh方法创建SpringMVC的组件
data:image/s3,"s3://crabby-images/742d1/742d16b17aecde1dd3302c27d06cbb0bf3df471f" alt="image-20211208175553011"
每个具体的 DispatcherServlet 创建的是一个 Child Context,代表一个独立的 IoC 容器;
ContextLoaderListener 所创建的是一个 Root Context,代表全局唯一的一个公共 IoC 容器。
四种方式获取当前容器
现在这句话应该可以清晰了:
WebApplicationContext是专门为web应用准备的,从WebApplicationContext中可以获得ServletContext的引用。
同时,整个Web应用上下文对象将作为域对象属性放置在Web容器的facade中,以便Web应用可以访问spring上下文。
data:image/s3,"s3://crabby-images/3d541/3d541ac967bf296903d925eec29cadf48c527f85" alt="image-20211208143204632"
1 | // ROOT XmlWebApplicationContext |
data:image/s3,"s3://crabby-images/a248e/a248e69af0838f27934d190c3e015af2b0d458b2" alt="image-20211208181647515"