Spring Boot如何启动嵌入式Tomcat?

一个BeanPostProcessor,为定制化嵌入式Web容器,在postProcessBeforeInitialization过程中去寻找Spring容器中WebServerFactoryCustomizer类型的Bean,并依次调用WebServerFactoryCustomizer接口的customize方法做一些定制化。

public interface WebServerFactoryCustomizer {

void customize(T factory);

}

创建、启动嵌入式Web容器

============================================================================

Spring的ApplicationContext,其抽象实现类AbstractApplicationContext#refresh

用来新建或刷新一个ApplicationContext,在refresh中会调用onRefresh,AbstractApplicationContext的子类可以重写onRefresh实现Context刷新逻辑。

因此重写 ServletWebServerApplicationContext#onRefresh 创建嵌入式Web容器:

重写onRefresh方法,调用createWebServer创建和启动Tomcat。

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:

注册Servlet

========================================================================

  • 有@RestController,为什么还要自己去注册Servlet给Tomcat?

可能有些场景需要注册你自己写的一个Servlet提供辅助功能,与主程序分开。

  • Sprong Boot 不注册Servlet 给Tomcat 直接用 @Controller 就能实现Servlet功能是为啥呢?

因为Sprong Boot默认给我们注册了DispatcherSetvlet。

Servlet注解


在Spring Boot启动类上加上 @ServletComponentScan 注解后,使用@WebServlet、@WebFilter、@WebListener标记的Servlet、Filter、Listener就可以自动注册到Servlet容器。

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

ServletRegistrationBean


Spring Boot提供了

  • ServletRegistrationBean

  • FilterRegistrationBean

  • ServletListenerRegistrationBean

分别用来注册Servlet、Filter、Listener。

假如要注册一个Servlet:

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

Spring会把这种类型的Bean收集起来,根据Bean里的定义向Tomcat注册Servlet。

动态注册


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

实现ServletContextInitializer接口的类会被spring管理,而不是被Servlet容器管理。

@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以后才有的功能。

通过 ServletContextInitializer 接口可以向 Web 容器注册 Servlet,实现 ServletContextInitializer 接口的Bean被speing管理,但是在什么时机触发其onStartup()方法的呢?

通过 Tomcat 中的 ServletContainerInitializer 接口实现者,如TomcatStarter,创建tomcat时设置了该类,在tomcat启动时会触发ServletContainerInitializer实现者的onStartup()方法,在这个方法中触发ServletContextInitializer接口的onStartup()方法,如注册DispatcherServlet。

DispatcherServletRegistrationBean实现了ServletContextInitializer接口,它的作用就是向Tomcat注册DispatcherServlet,那它是在什么时候、如何被使用的呢?

prepareContext方法调用了另一个私有方法configureContext,这个方法就包括了往Tomcat的Context添加ServletContainerInitializer对象:

context.addServletContainerInitializer(starter, NO_CLASSES);

其中有DispatcherServletRegistrationBean。

最后

小编精心为大家准备了一手资料

以上Java高级架构资料、源码、笔记、视频。Dubbo、Redis、设计模式、Netty、zookeeper、Spring cloud、分布式、高并发等架构技术

【附】架构书籍

  1. BAT面试的20道高频数据库问题解析
  2. Java面试宝典
  3. Netty实战
  4. 算法

BATJ面试要点及Java架构师进阶资料

1719171747185)]

[外链图片转存中…(img-bZ8E6AE7-1719171747186)]

以上Java高级架构资料、源码、笔记、视频。Dubbo、Redis、设计模式、Netty、zookeeper、Spring cloud、分布式、高并发等架构技术

【附】架构书籍

  1. BAT面试的20道高频数据库问题解析
  2. Java面试宝典
  3. Netty实战
  4. 算法

[外链图片转存中…(img-kLydlwys-1719171747186)]

BATJ面试要点及Java架构师进阶资料

[外链图片转存中…(img-q3GJcTnt-1719171747187)]

  • 27
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring Boot 启动内嵌 Tomcat 的过程可以分为以下几个步骤: 1. 配置 Spring Boot 应用程序的 pom.xml 文件,添加以下依赖项: ``` <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> ``` 2. 创建一个 Spring Boot 应用程序类,并在其 main 方法中调用 SpringApplication.run 方法,如下所示: ``` @SpringBootApplication public class MyApp { public static void main(String[] args) { SpringApplication.run(MyApp.class, args); } } ``` 3. 在应用程序配置文件 application.properties 或 application.yml 中配置 Tomcat 服务器相关的属性,例如端口号、上下文路径等,如下所示: ``` # application.properties server.port=8080 server.servlet.context-path=/myapp ``` ``` # application.yml server: port: 8080 servlet: context-path: /myapp ``` 4. 使用 Spring Boot 提供的嵌入式 Tomcat 作为 Web 服务器。Spring Boot 会根据应用程序的配置自动配置 Tomcat 服务器,创建 TomcatEmbeddedServletContainerFactory 对象,并将其注入到 Spring 容器中。 5. 在应用程序启动过程中,Spring Boot 会扫描应用程序中所有的 @Controller、@RestController、@RequestMapping 等注解,并将其注册到 Tomcat 服务器中。 6. 当应用程序收到请求时,Tomcat 服务器会将请求转发给对应的 Controller 方法,Controller 方法会处理请求并返回响应,Tomcat 服务器将响应发送给客户端。 以上就是 Spring Boot 启动内嵌 Tomcat 的过程。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值