
Fiddler 显示的 34ee 和 0 并非 Go 代码注入的异常内容,而是 HTTP/1.1 分块传输编码(Chunked Transfer Encoding)的标准格式标识——Fiddler 正确解析了该编码,而 Postman 默认自动解码,故未显式呈现。
fiddler 显示的 `34ee` 和 `0` 并非 go 代码注入的异常内容,而是 http/1.1 分块传输编码(chunked transfer encoding)的标准格式标识——fiddler 正确解析了该编码,而 postman 默认自动解码,故未显式呈现。
在 Go Web 开发中,当你使用标准 http.ResponseWriter 直接写入原始字节(如 w.Write(response)),且未显式设置 Content-Length 头时,Go 的 net/http 包会自动启用分块传输编码(Chunked Transfer Encoding),前提是满足以下条件:
- 使用 HTTP/1.1 协议(默认);
- 未设置 Content-Length 头;
- 响应体长度未知或流式生成(例如 ioutil.ReadAll 后一次性写入,但 Go 仍按“无长度声明”处理)。
此时,底层 HTTP 服务器会将响应体按块封装,每块以十六进制长度开头,后跟换行符、数据本身、再换行;结束块为 0\r\n\r\n。你看到的 34ee(十六进制) = 13550(十进制),恰好是你 JSON 响应体的字节数;末尾的 0 即终止块标识。
✅ 验证方式(终端 curl):
curl -i http://localhost:8080/api/data
若响应头含 Transfer-Encoding: chunked,即确认启用分块编码。
? 解决方案(推荐):
方案一:显式设置 Content-Length(最简洁、高效)
在 w.Write() 前添加长度头,禁用分块编码:
response, err := ioutil.ReadAll(resp.Body)
if err != nil {
http.Error(w, "Failed to read response", http.StatusInternalServerError)
return
}
w.Header().Set("Content-Type", "application/json; charset=UTF-8")
w.Header().Set("Content-Length", strconv.Itoa(len(response))) // ? 关键:预设长度
w.Write(response)⚠️ 注意:Content-Length 必须是精确字节数(非字符串长度),尤其注意 UTF-8 编码下中文字符占多字节。len(response) 返回的是字节长度,完全正确。
方案二:禁用分块编码(不推荐,仅作了解)
通过设置 w.Header().Set("Connection", "close") 可强制关闭连接、绕过分块,但违反 HTTP/1.1 持久连接原则,影响性能与兼容性,不建议生产环境使用。
? 为什么 Postman 看不到 34ee?
Postman 默认自动解码 Transfer-Encoding: chunked,直接展示解码后的原始响应体;而 Fiddler 作为底层抓包工具,默认显示未经处理的原始 HTTP 流,因此保留 chunk header,提示你“需解码”。点击 “Decode” 即可还原为干净 JSON —— 这恰恰说明协议行为完全正常,并非 Bug。
? 总结:
- 34ee 和 0 是合法、标准的 HTTP/1.1 分块编码标记,与 Go 代码无关,也非 Fiddler 错误;
- 根本解决方法是显式设置 Content-Length 头,既提升可预测性,又避免中间设备(如某些代理、CDN)对分块编码的兼容性问题;
- 所有 JSON API 响应建议统一设置 Content-Type + Content-Length,增强服务健壮性与调试友好性。










