对于Tomcat技术,Java程序员应该都非常熟悉,它是Web应用最常用的容器技术。我们最早的开发的项目基本都是部署在Tomcat下运行。
在软件开发的浪潮中,技术的更迭总是让人目不暇接。在SpringBoot框架中,我们使用最多的是Tomcat,这是SpringBoot默认的容器技术,而且是内嵌式的Tomcat。
但是,许多大厂的SpringBoot项目纷纷弃用Tomcat,转而拥抱Undertow。这背后究竟隐藏着什么秘密?今天,就让我们一起揭开这层神秘的面纱。
接下来先比较Tomcat与Undertow的优劣,我们从以下几个方面进行概述。
看重点
Tomcat的优势:
成熟与广泛使用:Tomcat是Apache基金会的开源项目,作为一个成熟的Web服务器和Servlet容器,它包含了HTTP服务器,因此也可以作为单独的Web服务器使用。
丰富的功能:Tomcat提供了管理和控制平台、安全局管理和Tomcat阀等功能,这些都是Tomcat特有的功能。
Undertow的优势:
高性能:Undertow基于非阻塞I/O和异步处理,能够提供更高的性能和更好的并发处理能力。它在高并发场景下尤其表现出色,能够处理更多请求并提供更快的响应速度。
轻量级:Undertow的代码库相对较小,这使得它在资源占用和启动时间方面具有优势,特别适合需要快速启动和低内存占用的应用场景。
更好的异步支持:Undertow提供了完善的异步支持,非常适合处理高并发的请求。
易于配置:Undertow提供了简单易用的配置选项,使得开发者可以快速定制和部署应用程序。Spring Boot中的自动配置功能使得切换到Undertow变得非常简单。
WebSocket支持:Undertow在WebSocket方面的实现更加轻量级和高效,这使得它在实时Web应用场景中更具优势。
HTTP/2支持:Undertow支持HTTP/2协议,开发者可以利用HTTP/2的性能优势,如头部压缩、服务器推送等。
总结:如果你的应用场景需要高性能、快速启动和轻量级的服务器,Undertow可能是更好的选择。而如果你需要一个成熟、功能丰富的Web服务器,且对资源占用和并发处理能力要求不是特别高,Tomcat则可能更适合你的需求。
大厂项目弃用Tomcat原因总结:在高并发系统中,Tomcat相对来说比较弱。在相同的机器配置下,模拟相等的请求数,Undertow在性能和内存使用方面都是最优的。并且Undertow新版本默认使用持久连接,这将会进一步提高它的并发吞吐能力。所以,如果是高并发的业务系统,Undertow是最佳选择。
一、什么是Undertow?
Undertow 是一个采用 Java 开发的灵活的高性能 Web 服务器,提供包括阻塞和基于 NIO 的非堵塞机制。Undertow 是红帽公司的开源产品,是 Wildfly 默认的 Web 服务器。Undertow 提供一个基础的架构用来构建 Web 服务器,这是一个完全为嵌入式设计的项目,提供易用的构建器 API,完全向下兼容 Java EE Servlet 3.1 和低级非堵塞的处理器。
Undertow的主要特点:
- 高性能 在多款同类产品的压测中,在高并发情况下表现出色。
- Servlet4.0 支持 它提供了对 Servlet4.0 的支持。
- Web Socket 完全支持,包括JSR-356,用以满足 Web 应用巨大数量的客户端。
- 内嵌式 它不需要容器,只需通过 API 即可快速搭建 Web 服务器。
- 灵活性 交由链式Handler配置和处理请求,可以最小化按需加载模块,无须加载多余功能。
- 轻量级 它是一个 内嵌Web 服务器, 由两个核心 Jar 包组成
二、SpringBoot标准项目中如何使用Undertow
1.pom.xml中引入依赖
<!-- 启动器依赖 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
<!-- 移除Tomcat的依赖 -->
<exclusions>
<exclusion>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-tomcat</artifactId>
</exclusion>
</exclusions>
</dependency>
<!-- 采用Undertow依赖 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-undertow</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
</dependency>
2.application.yml核心配置
server:
undertow:
# 设置IO线程数, 它主要执行非阻塞的任务,它们会负责多个连接, 默认设置每个CPU核心一个线程
io-threads: 2
# 阻塞任务线程池, 当执行类似servlet请求阻塞操作, undertow会从这个线程池中取得线程,它的值设置取决于系统的负载
worker-threads: 1000
# 以下的配置会影响buffer,这些buffer会用于服务器连接的IO操作,有点类似netty的池化内存管理
# 每块buffer的空间大小,越小的空间被利用越充分
buffer-size: 1024
# 是否分配的直接内存
direct-buffers: true
3.运行效果