【1】SpringMVC和Tomcat

项目背景是SpringMVC部署在外部的Tomcat8.5.382中。这里我们配置DispatcherServlet的​​<servlet-name>​​为DispatcherServlet

① ServletContext

如下图所示,这里获取到的全局上下文​​ServletContext​​​是​​ApplicationContextFacade​​​,是一个​​ApplicationContext​​的外观对象(也就是说这里应用了设计模式中的外观模式)。

​ApplicationContextFacade​​​有三个属性:​​classCache​​​、​​objectCache​​​和源目标对象context(​​ApplicationContext​​)。

源对象​​context​​(​​ApplicationContext​​) 就是我们平时使用的实际Tomcat应用上下文,其又拥有​​StandardContext​​​的引用。我们​​servletContext.setAttribute("testAttr","testAttrValue");​​​设置的属性就在​​attributes​​​下。
Spring、SpringMVC和SpringBoot框架中那些容器_spring

① attributes概览

这里可以看到我们的​​SpringMVC​​​容器以​​org.springframework.web.servlet.FrameworkServlet.CONTEXT.DispatcherServlet​​​为键存储在​​ApplicationContext​​​中的​​attributes​​​里面。
Spring、SpringMVC和SpringBoot框架中那些容器_容器_02

② attributes中的​​org.apache.catalina.jsp_classpath​

也就是我们常说的classpath,这里我们以分号换行分隔展示:

# 本地tomcat路径下的
/D:/softintall/apache-tomcat-8.5.38/lib/;
/D:/softintall/apache-tomcat-8.5.38/lib/annotations-api.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/catalina-ant.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/catalina-ha.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/catalina-storeconfig.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/catalina-tribes.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/catalina.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/ecj-4.6.3.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/el-api.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/jasper-el.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/jasper.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/jaspic-api.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/jsp-api.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/servlet-api.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/tomcat-api.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/tomcat-coyote.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/tomcat-dbcp.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/tomcat-i18n-es.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/tomcat-i18n-fr.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/tomcat-i18n-ja.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/tomcat-i18n-ru.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/tomcat-jdbc.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/tomcat-jni.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/tomcat-util-scan.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/tomcat-util.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/tomcat-websocket.jar;
/D:/softintall/apache-tomcat-8.5.38/lib/websocket-api.jar;
/D:/softintall/apache-tomcat-8.5.38/bin/bootstrap.jar;
/D:/softintall/apache-tomcat-8.5.38/bin/tomcat-juli.jar;

# idea路径下的
/C:/Users/12746/.IntelliJIdea2018.2/system/captureAgent/debugger-agent.jar;

# jre路径下的
/C:/Program Files/Java/jdk1.8.0_101/jre/lib/ext/access-bridge-64.jar;
/C:/Program Files/Java/jdk1.8.0_101/jre/lib/ext/cldrdata.jar;
/C:/Program Files/Java/jdk1.8.0_101/jre/lib/ext/dnsns.jar;
/C:/Program Files/Java/jdk1.8.0_101/jre/lib/ext/jaccess.jar;
/C:/Program Files/Java/jdk1.8.0_101/jre/lib/ext/jfxrt.jar;
/C:/Program Files/Java/jdk1.8.0_101/jre/lib/ext/localedata.jar;
/C:/Program Files/Java/jdk1.8.0_101/jre/lib/ext/nashorn.jar;
/C:/Program Files/Java/jdk1.8.0_101/jre/lib/ext/sunec.jar;
/C:/Program Files/Java/jdk1.8.0_101/jre/lib/ext/sunjce_provider.jar;
/C:/Program Files/Java/jdk1.8.0_101/jre/lib/ext/sunmscapi.jar;
/C:/Program Files/Java/jdk1.8.0_101/jre/lib/ext/sunpkcs11.jar;
/C:/Program Files/Java/jdk1.8.0_101/jre/lib/ext/zipfs.jar

② SpringMVC容器​​XmlWebApplicationContext​

这里项目创建的SpringMVC容器是​​XmlWebApplicationContext​​​,上面我们提到我们的​​SpringMVC​​​容器以​​org.springframework.web.servlet.FrameworkServlet.CONTEXT.DispatcherServlet​​​为键存储在​​ApplicationContext​​​中的​​attributes​​​里面,值是​​WebApplicationContext for namespace 'DispatcherServlet-servlet', started on Thu Oct 07 11:00:54 CST 2021​

org.springframework.web.servlet.FrameworkServlet.CONTEXT.DispatcherServlet=
WebApplicationContext for namespace 'DispatcherServlet-servlet', started on Thu Oct 07 11:00:54 CST 2021

Spring、SpringMVC和SpringBoot框架中那些容器_spring_03

① XmlWebApplicationContext对象概览

Spring、SpringMVC和SpringBoot框架中那些容器_spring mvc_04

  • 其拥有servletContext引用;
  • 其拥有servletConfig引用;
  • namespace为​​DispatcherServlet-servlet​​;
  • configLocations定义了配置文件路径,这里为​​classpath:springMVC.xml​​;
  • ​displayName=WebApplicationContext for namespace 'DispatcherServlet-servlet'​​;
  • 这里parent=null;

下图中​​mvc-attr​​​是​​org.springframework.web.servlet.FrameworkServlet.CONTEXT.​​​+servletName
Spring、SpringMVC和SpringBoot框架中那些容器_容器_05

【2】SSM项目部署在外部Tomcat

这里我们配置DispatcherServlet的​​<servlet-name>​​为springmvc。

① ServletContext

Spring、SpringMVC和SpringBoot框架中那些容器_tomcat_06
这里​​attributes​​中多了spring容器:

org.springframework.web.context.WebApplicationContext.ROOT=
Root WebApplicationContext: startup date [Fri Oct 08 20:49:42 CST 2021]; root of context hierarchy

Spring容器对象概览
Spring、SpringMVC和SpringBoot框架中那些容器_spring mvc_07

  • ​id=org.springframework.web.context.WebApplicationContext:​​;
  • ​displayName=Root WebApplicationContext​​;
  • ​parent=null​​;
  • ​configLocations​​​=​​/WEB-INF/classes/spring/applicationContext-*.xml​

springmvc容器以下面键值对储存在attributes中

org.springframework.web.servlet.FrameworkServlet.CONTEXT.springmvc=
WebApplicationContext for namespace 'springmvc-servlet': startup date [Fri Oct 08 20:49:48 CST 2021]; parent: Root WebApplicationContext

如下图所示,SpringMVC容器的parent属性指向了Spring容器。
Spring、SpringMVC和SpringBoot框架中那些容器_jar_08
也就是说,此时Tomcat的​​​ServletContext​​​的​​attributes​​​中存放了​​Spring容器​​​和​​SpringMVC容器​​​。其中​​SpringMVC容器的parent指向Spring容器​​​!
Spring、SpringMVC和SpringBoot框架中那些容器_容器_09

【3】SpringBoot下内置Tomcat

① ServletContext

Spring、SpringMVC和SpringBoot框架中那些容器_容器_10
这里需要注意的是此时attributes有两个key对应同一个容器对象:

org.springframework.web.context.WebApplicationContext.ROOT=
org.springframework.boot.web.servlet.context.AnnotationConfigServletWebServerApplicationContext@a668c39, started on Fri Oct 08 21:35:45 CST 2021

org.springframework.web.servlet.FrameworkServlet.CONTEXT.dispatcherServlet=
org.springframework.boot.web.servlet.context.AnnotationConfigServletWebServerApplicationContext@a668c39, started on Fri Oct 08 21:35:45 CST 2021

也就是说当前SpringBoot内置Tomcat环境下,只有​​ServletContext​​​和​​AnnotationConfigServletWebServerApplicationContext​​​。此时不存在Spring容器、SpringMVC容器,只有SpringBoot的容器(也可以理解为应用容器)
Spring、SpringMVC和SpringBoot框架中那些容器_容器_11

Spring、SpringMVC和SpringBoot框架中那些容器_容器_12

关于SpringMVC容器存储在ServletContext中的键

// 返回ServletContext 中 储存 servlet's WebApplicationContext的属性名
public String getServletContextAttributeName() {
return SERVLET_CONTEXT_PREFIX + getServletName();
}
// SERVLET_CONTEXT_PREFIX 如下
public static final String SERVLET_CONTEXT_PREFIX = FrameworkServlet.class.getName() + ".CONTEXT.";

getServletName()就是你指定的DispatcherServlet的名字,默认为dispatcherServlet