Go语言读取HTTP响应Body需及时关闭、完整读取并按格式解码:必须调用resp.Body.Close()(推荐defer),用io.ReadAll安全读取全部内容,JSON响应优先使用json.NewDecoder流式解析,注意编码处理与状态码检查。

Go语言中读取HTTP响应Body的关键是及时关闭Body并正确解码内容,否则可能引发内存泄漏或连接复用失败。
必须调用resp.Body.Close()
每次发起HTTP请求后,无论成功与否,都应确保关闭响应体。Go的HTTP客户端会复用底层TCP连接,而未关闭Body会导致连接无法释放,最终耗尽连接池。
- 推荐用
defer resp.Body.Close()放在if err != nil检查之后,避免空指针panic - 若需多次读取Body(如调试+解析),需先用
io.ReadAll一次性读出字节切片,再分别处理
用io.ReadAll安全读取全部内容
Body是io.ReadCloser接口,不能直接打印或重复读取。最稳妥的方式是用io.ReadAll转成[]byte:
-
bodyBytes, err := io.ReadAll(resp.Body)—— 简洁、自动处理EOF和错误 - 不建议用
bufio.Scanner或循环Read,除非流式处理超大响应(如文件下载) - 读取后可直接用
string(bodyBytes)查看,或传给json.Unmarshal等解析函数
JSON响应推荐用json.NewDecoder流式解析
对JSON API响应,更高效的方式是跳过中间字节切片,直接用json.NewDecoder解析Body:
立即学习“go语言免费学习笔记(深入)”;
-
err := json.NewDecoder(resp.Body).Decode(&data)—— 内存友好,适合大JSON - 注意:此方式会消耗Body,后续不能再读;如需日志原始内容,得先
io.ReadAll保存 - 记得在
Decode前检查resp.StatusCode,避免对4xx/5xx响应误解析
处理非UTF-8编码或含BOM的内容
某些API返回GBK、Shift-JIS等编码,或带UTF-8 BOM头,直接转string会乱码:
- 先用
charset.NewReaderLabel(来自golang.org/x/net/html/charset)自动识别编码 - 或手动检测BOM:
bytes.HasPrefix(bodyBytes, []byte{0xEF, 0xBB, 0xBF}),然后截掉 - 再用
strings.ToValidUTF8或第三方库(如go-runewidth)做兼容转换
基本上就这些。核心就三点:关Body、读全量、按格式解码。不复杂但容易忽略。










