Spring是如何运行在tomcat服务中的?

前言: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页面等,这3类标签应该最熟悉了,它们的一般配置:

    
      
        org.springframework.web.context.ContextLoaderListener  
      
      
        contextConfigLocation  
        classpath:spring/applicationContext.xml  
      
     
      
        DispatcherServlet
        org.springframework.web.servlet.DispatcherServlet  
        
          
            contextConfigLocation  
            classpath:spring/dispatcher-servlet.xml  
          
        1
      
      
        DispatcherServlet  
        /   
    

最主要的便是这里的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。

此处先给出一个参考链接:

SpringBoot内置Tomcat启动时间_串一串爪娃子-CSDN博客_springboot何时启动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项目。

此处先给出一个参考链接:

SpringBoot学习系列-war包在tomcat下启动原理 - 海绵般汲取 - 博客园

参考链接:

Tomcat原理系列之四:Tomat如何启动spring(加载web.xml)_2017年-CSDN博客_tomcat加载web.xml

SpringBoot内置Tomcat启动时间_串一串爪娃子-CSDN博客_springboot何时启动tomcat

SpringBoot学习系列-war包在tomcat下启动原理 - 海绵般汲取 - 博客园

java类加载: SPI机制及反射原理_东方小硕-CSDN博客_spi 反射

你可能感兴趣的:(tomcat,spring,spring,boot)