springboot的web服务器

我们知道,springboot是spring-mvc的整合,其中一项优点是内嵌服务器。但是,他并非一定要使用内嵌服务器,springboot也提供了外部部署的选项。

springboot启动

      • 内嵌服务器
      • 外部服务器

内嵌服务器

首先摘入官网的一段话:

Under the hood, Spring Boot uses a different type of ApplicationContext for embedded servlet container support. The ServletWebServerApplicationContext is a special type of WebApplicationContext that bootstraps itself by searching for a single ServletWebServerFactory bean. Usually a TomcatServletWebServerFactory, JettyServletWebServerFactory, or UndertowServletWebServerFactory has been auto-configured.

使用ServletWebServerApplicationContext 可以发现服务器。

那么这些服务器在哪里配置呢?按照springboot的套路,一定是在自动配置类中:

springboot的web服务器_第1张图片
他这里有tomcat,jetty,undertow几个选项。

springboot的web服务器_第2张图片
因为我导入了web场景启动器:

 <dependency>
        <groupId>org.springframework.bootgroupId>
        <artifactId>spring-boot-starter-webartifactId>
    dependency>

所以自带tomcat的包。

所以我们的容器中就会有TomcatServletWebServerFactory


那么,内嵌tomcat是如何启动的呢?我这里截几张图表示重要的点:

springboot的web服务器_第3张图片
内嵌容器一定是走main方法的。

springboot的web服务器_第4张图片

创建容器和刷新容器。我们进入refreshContext方法。

springboot的web服务器_第5张图片

springboot的web服务器_第6张图片
springboot的web服务器_第7张图片
进入getWebServerFactory

springboot的web服务器_第8张图片
他是按类型去容器拿的,当然,我们容器里有TomcatServletWebServerFactory,他就是个ServletWebServerFactory

springboot的web服务器_第9张图片
springboot的web服务器_第10张图片
再进入getSelfInitializer

springboot的web服务器_第11张图片
他返回的是这么一个东西:

@FunctionalInterface
public interface ServletContextInitializer {

	/**
	 * Configure the given {@link ServletContext} with any servlets, filters, listeners
	 * context-params and attributes necessary for initialization.
	 * @param servletContext the {@code ServletContext} to initialize
	 * @throws ServletException if any call against the given {@code ServletContext}
	 * throws a {@code ServletException}
	 */
	void onStartup(ServletContext servletContext) throws ServletException;

}

他接受一个参数,不返回值,像Consumer一样。这时是不会调用的,要等到调用ServletContextInitializer 类型的onStartup方法才会将ServletContext 传进来,传给selfInitialize使用。

ServletContextInitializer 这个东西相当于是spring自己来接管servlet的。

springboot的web服务器_第12张图片
getWebServer这里我们把tomcat给new出来了。

springboot的web服务器_第13张图片
然后走服务器的初始化。

springboot的web服务器_第14张图片
此时tomcat启动。

springboot的web服务器_第15张图片
然后把初始化器丢进去,走ServletContextInitializeronStartup方法。

springboot的web服务器_第16张图片
这是从IOC容器中取组件构建servlet环境。

外部服务器

使用外部服务器,比如tomcat,该怎么做呢?

首先,改造我们的代码:

    <packaging>warpackaging>
    <artifactId>springboot-contextartifactId>
<dependencies>
    <dependency>
        <groupId>org.springframework.bootgroupId>
        <artifactId>spring-boot-starter-webartifactId>
        <exclusions>
            <exclusion>
                <groupId>org.springframework.bootgroupId>
                <artifactId>spring-boot-starter-tomcatartifactId>
            exclusion>
        exclusions>
    dependency>

    <dependency>
        <groupId>javax.servletgroupId>
        <artifactId>javax.servlet-apiartifactId>
        <version>3.1.0version>
        <scope>providedscope>
    dependency>
dependencies>
    <build>
        <plugins>
            <plugin>
                <artifactId>maven-war-pluginartifactId>
                <version>3.0.0version>
            plugin>
        plugins>
    build>
project>

首先要求打成war包,然后去掉自带的tomcat,加入javax包,以及一个打包工具。

@SpringBootApplication
public class MyStarter extends SpringBootServletInitializer {

    @Override
    protected SpringApplicationBuilder configure(SpringApplicationBuilder builder) {
        return builder.sources(MyStarter.class);
    }
}

主配置类去掉main方法(那是走内嵌服务器的逻辑),继承SpringBootServletInitializer,重写configure方法,把主配置类丢进去。

springboot的web服务器_第17张图片
打成war包后丢到webapp目录,运行tomcat就能启动程序了。

关键是,为什么?

我们问的其实是:tomcat怎么发现我们的程序入口的(这里是MyStarter)?

MyStarter继承了一个SpringBootServletInitializer,这又是什么?

public abstract class SpringBootServletInitializer
 implements WebApplicationInitializer

他是一个WebApplicationInitializer

WebApplicationInitializer又是被谁发现的(或者说调用的)?

有这么一个重要的类:

@HandlesTypes(WebApplicationInitializer.class)
public class SpringServletContainerInitializer 
implements ServletContainerInitializer

他的注释很多。

大意是,实现了servlet3.0规范的服务器(我用的是tomcat8,那是实现了的)会去META-INF/services/ 下面找一个ServletContainerInitializer(SPI机制)。

我们有吗?有的。

springboot的web服务器_第18张图片

文件的内容是:

org.springframework.web.SpringServletContainerInitializer

服务器看你注解里传的什么:

@HandlesTypes(WebApplicationInitializer.class)

WebApplicationInitializer。然后他把所有的WebApplicationInitializer给你整来,连着ServletContext一起丢给你:

@Override
	public void onStartup(@Nullable Set<Class<?>> 
	webAppInitializerClasses, ServletContext servletContext)
			throws ServletException 

springboot的web服务器_第19张图片
然后循环调用WebApplicationInitializeronStartup方法。

springboot的web服务器_第20张图片
于是就会走到SpringBootServletInitializeronStartup方法,springboot由此便可以启动了。


关于servlet3.0以及注解化的spring-mvc,可以参考这个

你可能感兴趣的:(springboot,spring,boot,服务器,tomcat,SPI,源码)