Spring Boot 项目中Http 请求如何对响应体进行压缩

我们梳理一下 Spring Boot 项目中是如何对 HTTP 请求的响应体 (Response Body) 进行压缩。

(请注意:压缩是针对服务器返回给客户端的响应体,而不是客户端发给服务器的请求 URL 或请求体)。

核心思想:减少网络传输的数据量,加快客户端加载速度。


结论:

Spring Boot 默认开启 了对常见文本类型(如 JSON, HTML, CSS, JS)的响应体压缩(Gzip),通常我们不需要做任何配置!只需要确保:

  1. 响应体大小超过了默认阈值 (2048 bytes)。
  2. 响应 Content-Type 是默认支持压缩的类型之一 (如 application/json)。
  3. 客户端在请求头中声明了 Accept-Encoding: gzip (目前浏览器和工具默认都会加)。

下面是详细的解释和配置方法:

1. 压缩工作原理 (协商机制)

压缩过程是由客户端和服务器自动协商完成的,对Controller 代码透明:

  1. 客户端请求 (Request) :
    浏览器或 HTTP 客户端发送请求时,在请求头中携带 Accept-Encoding 字段,告知服务器它支持哪些压缩算法。

    复制代码
     GET /api/data HTTP/1.1
     Host: example.com
     Accept: application/json
     Accept-Encoding: gzip, deflate, br 
  2. 服务器响应 (Response) :
    Spring Boot(及其内嵌容器如 Tomcat)检查到:

    • 服务器开启了压缩功能。

    • 客户端 Accept-Encoding 中包含服务器支持的算法(如 gzip)。

    • 响应的 Content-Type 在配置的 mime-types 列表中。

    • 响应体的大小超过了配置的 min-response-size
      如果所有条件满足,服务器将:

    • 使用协商好的算法(如 gzip)压缩响应体。

    • 在响应头中添加 Content-Encoding 字段,告知客户端使用了何种算法。

    • 响应头中的 Content-Length 变为压缩后的大小。

    • 发送压缩后的数据。

      HTTP/1.1 200 OK
      Content-Type: application/json
      Content-Encoding: gzip
      Content-Length: 850 // <- 压缩后的大小 (例如原始大小是 3KB)
      ...
      [压缩后的二进制数据]

  3. 客户端接收 :
    客户端看到 Content-Encoding: gzip 后,自动对收到的响应体进行解压,再处理原始数据。

2. Spring Boot 配置

你可以在 application.propertiesapplication.yml 中进行配置和微调。

使用 application.properties:

properties 复制代码
# 1. 启用响应压缩 ( ⭐ 默认值: true )
#    如果你确定前端有反向代理(如 Nginx)在做压缩,可以在这里设为 false,避免重复压缩。
server.compression.enabled=true 

# 2. 配置需要压缩的内容类型 MIME types ( ⭐ 默认值已包含常见类型)
#    只有 Content-Type 匹配这里的列表,才会被考虑压缩。
#    默认值包括: text/html, text/xml, text/plain, text/css, text/javascript, 
#               application/javascript, application/json, application/xml 等等。
#    注意:不要添加已经压缩过的类型,如 image/jpeg, image/png, application/zip 等,重复压缩浪费 CPU 且效果差。
server.compression.mime-types=application/json,application/xml,text/html,text/plain,application/javascript,text/css

# 3. 触发压缩的最小响应体大小 ( ⭐ 默认值: 2048 bytes,即 2KB )
#    如果响应体小于此值,即使满足其他条件也不会压缩。
#    因为对小数据进行压缩的 CPU 开销可能大于节省的带宽,得不偿失。
#    单位是字节。
server.compression.min-response-size=1024 # 例如,改为 1KB

# (可选) 排除某些 User-Agent
# server.compression.excluded-user-agents=some-bad-client

使用 application.yml:

yaml 复制代码
server:
 compression:
   enabled: true # 默认 true
   mime-types: # 默认包含常见类型
     - application/json
     - application/xml
     - text/html
     - text/plain
     - application/javascript
     - text/css
   min-response-size: 1024 # 默认 2048 bytes

3. Controller 代码示例

你的 Controller 代码无需做任何修改!Spring Boot / 内嵌服务器会自动处理。

java 复制代码
 import org.springframework.web.bind.annotation.GetMapping;
 import org.springframework.web.bind.annotation.PostMapping;
 import org.springframework.web.bind.annotation.RequestBody;
 import org.springframework.web.bind.annotation.RestController;
 import java.util.List;
 import java.util.stream.Collectors;
 import java.util.stream.IntStream;

 @RestController
 public class MyController {

     // 模拟一个返回较大数据,用于测试 GET 请求的响应压缩
     @GetMapping("/api/users")
     public List<String> getUsers() {
        // 生成一个较大的列表,确保 JSON 序列化后大小超过 server.compression.min-response-size
         return IntStream.range(0, 1000)
                 .mapToObj(i -> "User Name - " + i + " with some description text here.")
                 .collect(Collectors.toList());
     }
     
      // 模拟一个 POST 请求,它的响应体同样会被压缩
      @PostMapping("/api/users/filter")
     public List<String> filterUsers(@RequestBody String filter) {
          // 假设过滤后仍然返回一个大数据
          return IntStream.range(0, 800)
                 .mapToObj(i -> "Filtered User Name - " + i + " for filter " + filter)
                 .collect(Collectors.toList());
     }
 }

当客户端(带上Accept-Encoding: gzip)请求 /api/users/api/users/filter 时,如果返回的 JSON 大小超过了 min-response-size,Spring Boot 就会自动返回 Content-Encoding: gzip 的压缩响应。

4. 如何验证?

使用浏览器的开发者工具 (F12) -> 网络 (Network) 面板:

  1. 刷新页面或触发 API 请求。
  2. 找到你的 API 请求记录。
  3. 点击该请求,查看 "标头" (Headers) -> "响应标头" (Response Headers)。
    • 如果看到 Content-Encoding: gzip (或 br, deflate),则表示压缩成功。
  4. 在请求列表的大小 (Size) 列,Chrome 等浏览器会显示两个值:
    • 上面的小数字:网络传输的压缩后大小。
    • 下面的大数字:解压后的原始大小。
    • 两者有显著差异就说明压缩生效了。

或者使用 curl 命令:

bash 复制代码
# -v 显示详细信息(包含头)
# --compressed 告诉 curl 自动请求并解压 (它会自动加上 Accept-Encoding: gzip, deflate 并根据 Content-Encoding 解压)
# -o /dev/null 不输出内容到屏幕
curl -v --compressed http://localhost:8080/api/users -o /dev/null

在 curl 的输出中查找 Response Headers 是否包含 Content-Encoding: gzip

5. 注意事项

  • 反向代理 (Reverse Proxy) : 在生产环境中,Spring Boot 应用前端经常会有 Nginx, Apache 或负载均衡器。这些反向代理通常也具备非常高效的压缩能力。最佳实践通常是在反向代理层(如 Nginx)统一处理压缩 ,而在 Spring Boot 中关闭压缩 (server.compression.enabled=false),以避免重复压缩和减轻应用服务器的 CPU 负担。
  • CPU 开销: 压缩会消耗服务器 CPU 资源。
  • 不要压缩已压缩内容 : 确保 mime-types 里不包含图片 (jpg, png, gif)、视频、zip 包等,它们本身就是压缩格式,再次压缩基本无效甚至可能增大体积,且浪费 CPU。
  • Streaming 响应 : 对于流式响应(例如 StreamingResponseBody 或 WebFlux 的 Flux),压缩机制依然有效,但实现方式略有不同(边生成边压缩边发送)。

总的来说,Spring Boot 提供了开箱即用的响应体压缩功能,通过简单的 server.compression.* 属性即可配置,无需修改业务代码。

相关推荐
程序员岳焱1 小时前
Java 与 MySQL 性能优化:Java 实现百万数据分批次插入的最佳实践
后端·mysql·性能优化
隆里卡那唔2 小时前
在dify中通过http请求neo4j时为什么需要将localhost变为host.docker.internal
http·docker·neo4j
麦兜*2 小时前
Spring Boot启动优化7板斧(延迟初始化、组件扫描精准打击、JVM参数调优):砍掉70%启动时间的魔鬼实践
java·jvm·spring boot·后端·spring·spring cloud·系统架构
KK溜了溜了2 小时前
JAVA-springboot 整合Redis
java·spring boot·redis
大只鹅2 小时前
解决 Spring Boot 对 Elasticsearch 字段没有小驼峰映射的问题
spring boot·后端·elasticsearch
ai小鬼头2 小时前
AIStarter如何快速部署Stable Diffusion?**新手也能轻松上手的AI绘图
前端·后端·github
天河归来2 小时前
使用idea创建springboot单体项目
java·spring boot·intellij-idea
~山有木兮3 小时前
LiteHub中间件之限流实现
网络·http·中间件
IT_10243 小时前
Spring Boot项目开发实战销售管理系统——数据库设计!
java·开发语言·数据库·spring boot·后端·oracle
bobz9653 小时前
动态规划
后端