Spring Boot如何使用内嵌式的Tomcat和Jetty?

本文详细介绍了SpringBoot如何通过WebServer、ServletWebServerFactory和相关接口,实现对内嵌式Web容器如Tomcat和Jetty的抽象定制,包括注册Servlet、定制Web服务器参数及使用WebServerFactoryCustomizer进行高级定制。

Spring Boot 中 Web 容器相关的接口

既然要支持多种 Web 容器,Spring Boot 对内嵌式 Web 容器进行了抽象,定义了 WebServer 接口:

public interface WebServer {
    void start() throws WebServerException;
    void stop() throws WebServerException;
    int getPort();
}

各种 Web 容器比如 Tomcat 和 Jetty 需要去实现这个接口。Spring Boot 还定义了一个工厂 ServletWebServerFactory 来创建 Web 容器,返回的对象就是上面提到的 WebServer。

public interface ServletWebServerFactory {
    WebServer getWebServer(ServletContextInitializer... initializers);
}

可以看到 getWebServer 有个参数,类型是 ServletContextInitializer。它表示 ServletContext 的初始化器,用于 ServletContext 中的一些配置:

public interface ServletContextInitializer {
    void onStartup(ServletContext servletContext) throws ServletException;
}

这里请注意,上面提到的 getWebServer 方法会调用 ServletContextInitializer 的 onStartup 方法,也就是说如果你想在 Servlet 容器启动时做一些事情,比如注册你自己的 Servlet,可以实现一个 ServletContextInitializer,在 Web 容器启动时,Spring Boot 会把所有实现了 ServletContextInitializer 接口的类收集起来,统一调它们的 onStartup 方法。为了支持对内嵌式 Web 容器的定制化,Spring Boot 还定义了 WebServerFactoryCustomizerBeanPostProcessor 接口,它是一个 BeanPostProcessor,它在 postProcessBeforeInitialization 过程中去寻找 Spring 容器中 WebServerFactoryCustomizer 类型的 Bean,并依次调用 WebServerFactoryCustomizer 接口的 customize 方法做一些定制化。

public interface WebServerFactoryCustomizer<T extends WebServerFactory> {
    void customize(T factory);
}

内嵌式 Web 容器的创建和启动

铺垫了这些接口,我们再来看看 Spring Boot 是如何实例化和启动一个 Web 容器的。我们知道,Spring 的核心是一个 ApplicationContext,它的抽象实现类 AbstractApplicationContext 实现了著名的 refresh 方法,它用来新建或者刷新一个 ApplicationContext,在 refresh 方法中会调用 onRefresh 方法,AbstractApplicationContext 的子类可以重写这个 onRefresh 方法,来实现特定 Context 的刷新逻辑,因此 ServletWebServerApplicationContext 就是通过重写 onRefresh 方法来创建内嵌式的 Web 容器,具体创建过程是这样的:


@Override
protected void onRefresh() {
     super.onRefresh();
     try {
        //重写onRefresh方法,调用createWebServer创建和启动Tomcat
        createWebServer();
     }
     catch (Throwable ex) {
     }
}

//createWebServer的具体实现
private void createWebServer() {
    //这里WebServer是Spring Boot抽象出来的接口,具体实现类就是不同的Web容器
    WebServer webServer = this.webServer;
    ServletContext servletContext = this.getServletContext();
    
    //如果Web容器还没创建
    if (webServer == null && servletContext == null) {
        //通过Web容器工厂来创建
        ServletWebServerFactory factory = this.getWebServerFactory();
        //注意传入了一个"SelfInitializer"
        this.webServer = factory.getWebServer(new ServletContextInitializer[]{this.getSelfInitializer()});
        
    } else if (servletContext != null) {
        try {
            this.getSelfInitializer().onStartup(servletContext);
        } catch (ServletException var4) {
          ...
        }
    }

    this.initPropertySources();
}

再来看看 getWebServer 具体做了什么,以 Tomcat 为例,主要调用 Tomcat 的 API 去创建各种组件


public WebServer getWebServer(ServletContextInitializer... initializers) {
    //1.实例化一个Tomcat,可以理解为Server组件。
    Tomcat tomcat = new Tomcat();
    
    //2. 创建一个临时目录
    File baseDir = this.baseDirectory != null ? this.baseDirectory : this.createTempDir("tomcat");
    tomcat.setBaseDir(baseDir.getAbsolutePath());
    
    //3.初始化各种组件
    Connector connector = new Connector(this.protocol);
    tomcat.getService().addConnector(connector);
    this.customizeConnector(connector);
    tomcat.setConnector(connector);
    tomcat.getHost().setAutoDeploy(false);
    this.configureEngine(tomcat.getEngine());
    
    //4. 创建定制版的"Context"组件。
    this.prepareContext(tomcat.getHost(), initializers);
    return this.getTomcatWebServer(tomcat);
}

你可能好奇 prepareContext 方法是做什么的呢?这里的 Context 是指 Tomcat 中的 Context 组件,为了方便控制 Context 组件的行为,Spring Boot 定义了自己的 TomcatEmbeddedContext,它扩展了 Tomcat 的 StandardContext:

class TomcatEmbeddedContext extends StandardContext {}

注册 Servlet 的三种方式

1. Servlet 注解在 Spring Boot 启动类上加上 @ServletComponentScan 注解后,使用 @WebServlet、@WebFilter、@WebListener 标记的 Servlet、Filter、Listener 就可以自动注册到 Servlet 容器中,无需其他代码,我们通过下面的代码示例来理解一下。

@SpringBootApplication
@ServletComponentScan
public class xxxApplication
{}


@WebServlet("/hello")
public class HelloServlet extends HttpServlet {}

在 Web 应用的入口类上加上 @ServletComponentScan,并且在 Servlet 类上加上 @WebServlet,这样 Spring Boot 会负责将 Servlet 注册到内嵌的 Tomcat 中。

2. ServletRegistrationBean

同时 Spring Boot 也提供了 ServletRegistrationBean、FilterRegistrationBean 和 ServletListenerRegistrationBean 这三个类分别用来注册 Servlet、Filter、Listener。假如要注册一个 Servlet,可以这样做:

@Bean
public ServletRegistrationBean servletRegistrationBean() {
    return new ServletRegistrationBean(new HelloServlet(),"/hello");
}

这段代码实现的方法返回一个 ServletRegistrationBean,并将它当作 Bean 注册到 Spring 中,因此你需要把这段代码放到 Spring Boot 自动扫描的目录中,或者放到 @Configuration 标识的类中。

3. 动态注册

你还可以创建一个类去实现前面提到的 ServletContextInitializer 接口,并把它注册为一个 Bean,Spring Boot 会负责调用这个接口的 onStartup 方法。

@Component
public class MyServletRegister implements ServletContextInitializer {

    @Override
    public void onStartup(ServletContext servletContext) {
    
        //Servlet 3.0规范新的API
        ServletRegistration myServlet = servletContext
                .addServlet("HelloServlet", HelloServlet.class);
                
        myServlet.addMapping("/hello");
        
        myServlet.setInitParameter("name", "Hello Servlet");
    }

}

这里请注意两点:ServletRegistrationBean 其实也是通过 ServletContextInitializer 来实现的,它实现了 ServletContextInitializer 接口。注意到 onStartup 方法的参数是我们熟悉的 ServletContext,可以通过调用它的 addServlet 方法来动态注册新的 Servlet,这是 Servlet 3.0 以后才有的功能。

Web 容器的定制

第一种方式是通过通用的 Web 容器工厂 ConfigurableServletWebServerFactory,来定制一些 Web 容器通用的参数:

@Component
public class MyGeneralCustomizer implements
  WebServerFactoryCustomizer<ConfigurableServletWebServerFactory> {
  
    public void customize(ConfigurableServletWebServerFactory factory) {
        factory.setPort(8081);
        factory.setContextPath("/hello");
     }
}

第二种方式是通过特定 Web 容器的工厂比如 TomcatServletWebServerFactory 来进一步定制。下面的例子里,我们给 Tomcat 增加一个 Valve,这个 Valve 的功能是向请求头里添加 traceid,用于分布式追踪。TraceValve 的定义如下:

class TraceValve extends ValveBase {
    @Override
    public void invoke(Request request, Response response) throws IOException, ServletException {

        request.getCoyoteRequest().getMimeHeaders().
        addValue("traceid").setString("1234xxxxabcd");

        Valve next = getNext();
        if (null == next) {
            return;
        }

        next.invoke(request, response);
    }

}

跟第一种方式类似,再添加一个定制器,代码如下:

@Component
public class MyTomcatCustomizer implements
        WebServerFactoryCustomizer<TomcatServletWebServerFactory> {

    @Override
    public void customize(TomcatServletWebServerFactory factory) {
        factory.setPort(8081);
        factory.setContextPath("/hello");
        factory.addEngineValves(new TraceValve() );

    }
}

Spring Boot 实现内嵌 Tomcat 的核心机制是通过 **自动装配(Auto-Configuration)** **工厂模式**,结合 `ServletWebServerApplicationContext` 来完成的。其目标是在不依赖外部 Web 容器的情况下,将 Web 服务器(如 TomcatJetty、Undertow)直接集成到应用中。 --- ## ✅ Spring Boot 启动内嵌 Tomcat 的流程 ### 1. 启动类使用 `@SpringBootApplication` ```java @SpringBootApplication public class Application { public static void main(String[] args) { SpringApplication.run(Application.class, args); } } ``` ### 2. `SpringApplication.run()` 内部逻辑 `SpringApplication.run()` 方法会创建一个合适的 `ApplicationContext`,具体逻辑如下: #### 推断 Web 应用类型: ```java webApplicationType = WebApplicationType.deduceFromClasspath(); ``` 如果类路径下存在 `javax.servlet.Servlet` `org.springframework.web.context.ConfigurableWebApplicationContext`,则认为是 Servlet Web 应用。 #### 创建对应的 ApplicationContext: ```java context = createApplicationContext(); ``` 对于 Servlet Web 应用,会创建 `AnnotationConfigWebServerApplicationContext`。 --- ### 3. 刷新上下文时启动 Web 服务器 在 `refreshContext()` 过程中,最终调用 `onRefresh()` 方法,在 `ServletWebServerApplicationContext` 中实现如下: ```java @Override protected void onRefresh() { super.onRefresh(); try { createWebServer(); // 关键方法:创建 Web 服务器 } catch (Throwable ex) { throw new ApplicationContextException("Failed to start bean", ex); } } ``` --- ### 4. `createWebServer()` 方法详解 该方法负责创建内嵌的 Web 服务器(TomcatJetty 等)。 ```java private void createWebServer() { WebServer webServer = this.webServer; ServletContext servletContext = this.servletContext; if (webServer == null && servletContext == null) { // 获取 WebServerFactory(TomcatServletWebServerFactory) ServletWebServerFactory factory = getWebServerFactory(); // 创建 WebServer(TomcatWebServer) this.webServer = factory.getWebServer(getSelfInitializer()); } else if (servletContext != null) { try { getSelfInitializer().onStartup(servletContext); } catch (ServletException ex) { throw new ApplicationContextException("Cannot initialize servlet context", ex); } } initPropertySources(); } ``` --- ## ✅ `TomcatServletWebServerFactory` 是如何被选中的? Spring Boot 使用条件注解来决定使用哪个 WebServerFactory。 例如在 `TomcatServletWebServerFactory` 上有如下注解: ```java @ConditionalOnClass({ Servlet.class, Tomcat.class }) @ConditionalOnMissingBean(value = ServletWebServerFactory.class, search = SearchStrategy.CURRENT) public class TomcatServletWebServerFactory extends AbstractServletWebServerFactory implements ServletWebServerFactory {} ``` 只有当类路径中有 `Tomcat.class` `Servlet.class` 时,才会加载这个 Factory。 --- ## ✅ `TomcatWebServer` 的创建过程 ```java public WebServer getWebServer(ServletContextInitializer... initializers) { Tomcat tomcat = new Tomcat(); File baseDir = createTempDir("tomcat"); tomcat.setBaseDir(baseDir.getAbsolutePath()); // 设置 Connector(HTTP 端口等) Connector connector = new Connector(this.protocol); tomcat.setConnector(connector); // 添加 Context(Servlet 上下文) Context context = tomcat.addContext("", baseDir.getAbsolutePath()); // 注册 ServletContextInitializer(如 DispatcherServlet) for (ServletContextInitializer initializer : initializers) { initializer.onStartup(context.getServletContext()); } // 启动 Tomcat tomcat.start(); // 返回 WebServer 实例 return new TomcatWebServer(tomcat, getPort(), shutdown); } ``` --- ## ✅ 总结:Spring Boot 内嵌 Tomcat 的关键点 | 步骤 | 说明 | |------|------| | 1. 推断 Web 类型 | 根据类路径判断是否为 Web 应用 | | 2. 创建对应上下文 | 如 `ServletWebServerApplicationContext` | | 3. 刷新上下文时创建 WebServer | 调用 `createWebServer()` | | 4. 获取 WebServerFactory | 根据类路径选择 `TomcatServletWebServerFactory` | | 5. 创建 Tomcat 实例 | 配置 Connector、Context 并启动 | | 6. 返回 WebServer 实例 | 最终返回 `TomcatWebServer` | --- ## ✅ 示例:查看默认端口配置 你可以在 `application.properties` 中设置端口: ```properties server.port=8081 ``` Spring Boot 会自动读取该配置,并传递给 `TomcatServletWebServerFactory`。 --- ## ✅ 扩展:支持 Jetty 或 Undertow 只需修改依赖即可切换 Web 服务器: #### Jetty 示例: ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> <exclusions> <exclusion> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-tomcat</artifactId> </exclusion> </exclusions> </dependency> <dependency> <groupId>org.eclipse.jetty</groupId> <artifactId>jetty-server</artifactId> </dependency> ``` Spring Boot 会自动检测并使用 Jetty。 --- ###
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Thomas.Sir

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值