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 的连接参数(如最大连接数、超时时间)感兴趣,也可以继续深入!

相关推荐
tb_first1 小时前
万字超详细苍穹外卖学习笔记5
java·数据库·spring boot·笔记·学习·spring
Hx_Ma161 小时前
SpringBoot消息转换器扩展fastjson
java·spring boot·spring
Coder_preston1 小时前
Spring/Spring Boot实战:从入门到项目部署
java·spring boot·spring
belldeep1 小时前
Java:Tomcat 9 和 mermaid.min.js 10.9 上传.csv文件实现 Markdown 中 Mermaid 图表的渲染
java·tomcat·mermaid·去除flexmark
lang201509281 小时前
Tomcat Maven插件:部署与卸载的架构设计
java·tomcat·maven
lang201509282 小时前
Tomcat Maven插件全解析:开发部署一体化
java·tomcat·maven
sin22012 小时前
WebRTC--流程
spring boot·webrtc
tb_first3 小时前
万字超详细苍穹外卖学习笔记3
java·jvm·笔记·学习·spring·tomcat·maven
qq_12498707533 小时前
基于html的书城阅读器系统的设计与实现(源码+论文+部署+安装)
前端·vue.js·spring boot·后端·mysql·信息可视化·html
凤山老林3 小时前
SpringBoot + MyBatis-Plus 如何高效实现数据变更记录
java·spring boot·mybatis