前言:Tomcat是一类Web应用服务器,Spring 框架是一个轻量级的 Java 开发框架。Tomcat提供Web容器服务,Spring为应用开发提供平台。Web应用服务器有多种选择,开发框架也是如此。即Tomcat与Spring不存在依赖上的相互关系。
从SSM等组合到SpringBoot体系脚手架,Tomcat也从外部变为可使用内置,那么Spring是如何运行在Tomcat中的呢,启动Tomcat时,是如何加载Spring容器的,或者Spring容器又是如何启动Tomcat的呢?
1.Tomcat如何启动Spring容器的?
如前言所述,Spring与Tomcat不存在强依赖的相互关系,这里标题为Tomcat如何启动Spring容器的,故是Tomcat首先启动,然后加载Spring容器的:像Tomat+Spring+SpringMvc + 一类的组合便是这种方式。此类应用的部署方式大致是怎么样的呢?配置web.xml,打war包,放置在webApp包下面,启动tomcat等。
web.xml有哪些信息需要配置?除了404,welcome页面等,<filter>,<listener>,<servlet>这3类标签应该最熟悉了,它们的一般配置:
<!--启用spring--> <listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener- class> </listener> <context-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:spring/applicationContext.xml</param-value> </context-param> <!--启用springmvc,配置DispatcherServlet--> <servlet> <servlet-name>DispatcherServlet</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <init-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:spring/dispatcher-servlet.xml</param-value> </init-param> <load-on-startup>1</load-on-startup> </servlet> <servlet-mapping> <servlet-name>DispatcherServlet</servlet-name> <url-pattern>/</url-pattern> <!--会拦截URL中带“/”的请求。--> </servlet-mapping>
最主要的便是这里的listener与servlet配置。
tomcat启动时会加载和解析web.xml文件,获取其中配置的filter,servlet,listener等设置到context中后,接下来就是执行了。执行到配置的ContextLoaderListener时,该监听器最终会调用AbstractApplicationContext.refresh()便会启动Spring容器(具体流程可查看源码及参考文末参考链接),而通过加载DispatcherServlet,便可以将相应的web请求转发到DispatcherServlet,进而web服务便可使用Spring容器提供的服务了,其是流程上沟通二者的桥梁。
此处先给出一个参考链接:Tomcat原理系列之四:Tomat如何启动spring(加载web.xml)_2017年-CSDN博客_tomcat加载web.xml
2.SpringBoot内置的Tomcat是如何启动的?
SpringBoot 启动类main方法:执行SpringApplication的SpringApplication.run()方法,在准备上下文容器ApplicationContext时,获取的实例为AnnotationConfigServletWebServerApplicationContext,其继承ServletWebServerApplicationContext,并将AbstractApplicationContext启动Spring容器的refresh方法中的onRefresh()方法重写:添加creatWebServer方法,该方法会启动tomcat容器。所以SpringBoot在加载Spring容器时启动了tomcat。
此处先给出一个参考链接:
3.通过War包部署的SpringBoot又是如何启动的?
spring boot war启动是利用Servlet 3.0新增的ServletContainerInitializer接口结合SPI(Service Provider Interface)机制实现的。
1.springboot打war包,启动类继承:SpringServletContainerInitializer,打包将会Spring在spring-web-version.jar/META-INF/services/javax.servlet.ServletContainerInitializer文件中,配置了spring对ServletContainerInitializer接口的实现类org.springframework.web.SpringServletContainerInitializer。
2.Servlet Container启动阶段扫描jar包中META-INF/services/javax.servlet.ServletContainerInitializer文件,(从而达到无web.xml配置的启动)
获取ServletContainerInitializer实现类并实例化,解析ServletContainerInitializer上@HandlesTypes注解,
查找出@HandlesTypes限定的类型集合,作为ServletContainerInitializer.onStartup方法处理的第一个参数c。
3.Servlet Container依次调用每个ServletContainerInitializer实例的onStartup。war包启动的场景中会调用SpringServletContainerInitializer.onStartup方法,该方法循环调用c集合中每个 WebApplicationInitializer子类(即SpringBootServletInitializer)的onStartup方法。4.SpringBootServletInitializer.onStartup方法调用SpringBootServletInitializer.createRootApplicationContext方法,createRootApplicationContext方法中构建SpringApplication并执行SpringApplication.run方法以启动整个spring项目。此处先给出一个参考链接:
参考链接:
Tomcat原理系列之四:Tomat如何启动spring(加载web.xml)_2017年-CSDN博客_tomcat加载web.xml
SpringBoot内置Tomcat启动时间_串一串爪娃子-CSDN博客_springboot何时启动tomcat