答案:在Golang中读取网络响应需正确处理resp.Body并及时关闭。小数据可用io.ReadAll一次性读取;文本流适合bufio.Scanner逐行处理;JSON响应推荐json.NewDecoder直接解码;无论何种方式都必须defer resp.Body.Close(),并检查StatusCode与设置超时,确保资源释放与程序健壮性。

在Golang中读取网络响应内容,通常是在发送HTTP请求后从*http.Response中获取返回的数据。关键在于正确处理响应体(Body)并及时关闭资源,避免内存泄漏。下面介绍几种常见的实践方式。
使用ioutil.ReadAll一次性读取
最简单的方式是使用ioutil.ReadAll将响应体全部读入内存。适用于响应数据较小的场景。
- Go 1.16之后推荐使用
io.ReadAll替代ioutil.ReadAll。 - 大响应体可能导致内存占用过高。
示例代码:
resp, err := http.Get("https://api.example.com/data")
if err != nil {
log.Fatal(err)
}
defer resp.Body.Close()
body, err := io.ReadAll(resp.Body)
if err != nil {
log.Fatal(err)
}
fmt.Println(string(body))
使用bufio逐行读取
当响应是文本格式(如日志流、JSON行等),可使用bufio.Scanner逐行处理,节省内存。
立即学习“go语言免费学习笔记(深入)”;
适合处理大型文本响应或流式数据。
示例:
resp, err := http.Get("https://api.example.com/stream")
if err != nil {
log.Fatal(err)
}
defer resp.Body.Close()
scanner := bufio.NewScanner(resp.Body)
for scanner.Scan() {
fmt.Println(scanner.Text())
}
if err := scanner.Err(); err != nil {
log.Fatal(err)
}
直接解码JSON响应
如果响应是JSON格式,建议直接用json.NewDecoder解码,无需先读字符串。
这种方式更高效,尤其适合结构化数据。
示例:
type User struct {
Name string `json:"name"`
Age int `json:"age"`
}
resp, err := http.Get("https://www.php.cn/link/d3edfaac7b134307771e928cc3293131")
if err != nil {
log.Fatal(err)
}
defer resp.Body.Close()
var user User
if err := json.NewDecoder(resp.Body).Decode(&user); err != nil {
log.Fatal(err)
}
fmt.Printf("%+v\n", user)
处理错误与资源释放
无论采用哪种读取方式,都必须确保resp.Body.Close()被调用,防止连接未释放。
常见注意事项:
- 即使请求失败(如404、500),只要返回了
*http.Response,就需关闭Body。 - 网络错误或超时应设置
http.Client的超时时间。 - 检查
resp.StatusCode判断是否为预期响应。
增强版请求示例:
client := &http.Client{Timeout: 10 * time.Second}
req, _ := http.NewRequest("GET", "https://api.example.com/data", nil)
resp, err := client.Do(req)
if err != nil {
log.Fatal(err)
}
defer resp.Body.Close()
if resp.StatusCode != http.StatusOK {
log.Fatalf("HTTP %d", resp.StatusCode)
}
body, _ := io.ReadAll(resp.Body)
fmt.Println(string(body))
基本上就这些。根据响应大小和数据格式选择合适的读取方式,始终记得关闭Body,保证程序健壮性。










