Spring Boot 内嵌 Tomcat 处理 HTTP 请求的全过程

我们来完整、准确地梳理一遍 Spring Boot 内嵌 Tomcat 处理 HTTP 请求的全过程,并澄清 I/O 模型的问题。


✅ 完整流程(从客户端到响应返回)

假设你用 java -jar myapp.jar 启动了一个 Spring Boot 应用(内嵌 Tomcat),浏览器发起请求:

1️⃣ 客户端 → 服务器:建立 TCP 连接

  • 客户端(如浏览器)向服务器 IP:8080 发起 TCP 三次握手
  • 成功后,TCP 连接建立(此时还没有任何 HTTP 数据

2️⃣ 客户端发送 HTTP 请求数据

  • 在已建立的 TCP 连接上,客户端发送 原始 HTTP 报文 (字节流),例如:

    复制代码
    GET /api/user/1 HTTP/1.1
    Host: localhost:8080
    Accept: application/json

3️⃣ Tomcat 接收并解析 HTTP 请求

关键点:Tomcat 是 Servlet 容器,负责底层网络通信和 HTTP 协议解析

  • Tomcat 的 连接器(Connector) 监听 8080 端口
  • 接收 TCP 字节流解析成 HttpServletRequest / HttpServletResponse 对象
  • 这一步与 Spring 无关,是 Tomcat 自己完成的(基于 Servlet 规范)

🔧 I/O 模型问题(BIO / NIO)

  • 传统 Tomcat(老版本默认):BIO(Blocking I/O)→ 每个请求一个线程,性能差
  • 现代 Spring Boot 内嵌 Tomcat(默认)NIO(Non-blocking I/O) → 少量线程处理大量连接(基于 Java NIO 的 Selector 机制)
  • Tomcat 10+ 还支持 NIO2(AIO)和 APR(本地库)

所以你说的"bno"很可能是 "BIO" 的误打,而实际情况是:现代 Spring Boot 默认用的是 NIO!

4️⃣ 请求交给 Spring MVC 处理

  • Tomcat 将 HttpServletRequest 交给 Spring 的 DispatcherServlet
  • Spring MVC 开始工作:
    • HandlerMapping :找到对应的 @Controller 方法
    • HandlerAdapter :调用你的业务方法(如 @GetMapping("/api/user/{id}")
    • 参数解析:路径变量、JSON 转对象(使用 Jackson 等)
    • 执行你的业务逻辑(Service、DAO、Redis、DB 等)

5️⃣ 构建 HTTP 响应

  • 你的 Controller 方法返回对象(如 User
  • Spring 用 HttpMessageConverter (如 MappingJackson2HttpMessageConverter)将其序列化为 JSON 字符串
  • 写入 HttpServletResponse 的输出流

6️⃣ Tomcat 发送响应 + 断开连接

  • Tomcat 将响应组装成 HTTP 报文(含状态行、Header、Body)
  • 通过同一个 TCP 连接发送回客户端
  • 根据 Connection 头决定是否保持连接 (HTTP/1.1 默认 keep-alive):
    • 如果是 短连接 (或客户端发了 Connection: close)→ 四次挥手断开 TCP
    • 如果是 长连接 → 连接保持,等待下一个请求(或超时后关闭)

📌 关键澄清

你提到的说法 实际情况
"Tomcat 解析 HTTP 请求" 正确,这是 Tomcat 的核心职责
"其实是 bno" ❌ 应为 NIO (非阻塞 I/O),不是 BIO(阻塞 I/O)
Spring Boot 默认 I/O 模型 NIO (通过内嵌 Tomcat 的 NioEndpoint 实现)

💡 你可以通过日志确认:

Spring Boot 启动时会打印:
Tomcat initialized with port(s): 8080 (http)

并且底层使用的是 org.apache.tomcat.util.net.NioEndpoint


✅ 总结全流程(简化版)

复制代码
客户端
  │
  ▼
TCP 三次握手(建立连接)
  │
  ▼
发送 HTTP 请求字节流
  │
  ▼
Tomcat (NIO) 接收 → 解析为 HttpServletRequest
  │
  ▼
Spring MVC DispatcherServlet 分发请求
  │
  ▼
执行 @Controller 业务逻辑
  │
  ▼
返回结果 → 序列化为 JSON
  │
  ▼
Tomcat 写入 HttpServletResponse → 发送 HTTP 响应
  │
  ▼
根据 Connection 头决定:
   ├─ keep-alive → 保持 TCP 连接(长连接)
   └─ close     → TCP 四次挥手断开(短连接)

如果你对 NIO 的工作原理 (Selector、Channel、Buffer)或者 Spring Boot 如何配置 Tomcat 的连接参数(如最大连接数、超时时间)感兴趣,也可以继续深入!

相关推荐
逍遥德15 分钟前
SpringBoot数据库连接池HikariCP,Druid,Tomcat JDBC,DBCP2,c3p0配置使用
数据库·spring boot·tomcat
学术阿凡提34 分钟前
Spring Boot 集成 Fastjson2 完整教程:从入门到避坑
spring boot·安全·json
冷小鱼1 小时前
MyBatis 与 MyBatis-Plus:从入门到精通的完整指南
java·tomcat·mybatis
也许明天y2 小时前
LangChain4j + Spring Boot 多智能体协调架构原理深度解析
spring boot·后端·agent
阿丰资源3 小时前
基于Spring Boot的新闻推荐系统(源码+数据库+文档)
数据库·spring boot·后端
身如柳絮随风扬4 小时前
Spring Boot + Spring Cloud 集成 Elasticsearch:从零搭建企业级搜索服务
spring boot·elasticsearch·spring cloud
野生技术架构师5 小时前
Tomcat Service的设计和实现:StandardService
java·tomcat
流觞 无依5 小时前
Spring Boot 未授权访问漏洞排查与修复指南
java·spring boot·后端
Java开发的小李5 小时前
SpringBoot 高流量高并发 基础全面讲解
java·spring boot·后端·性能优化
极创信息6 小时前
信创领域五种主流CPU架构(X86 / ARM / RISC-V / MIPS / LoongArch)
java·arm开发·数据库·spring boot·mysql·软件工程·risc-v