答案:处理HTTP响应需检查StatusCode并关闭Body。首先判断resp.StatusCode是否为http.StatusOK,非200时应按状态码分类处理;无论状态如何,都需defer resp.Body.Close()防止泄漏;可封装checkResponse函数统一解析错误,提升健壮性。

在Go语言中处理HTTP响应状态错误,关键在于及时检查响应的Status和StatusCode,并正确释放资源。很多开发者只关注请求是否发送成功,却忽略了对响应状态的判断,导致程序在服务端出错时仍继续解析数据,引发更严重的问题。
检查响应状态码
使用http.Get或http.Client.Do发起请求后,必须检查返回的resp.StatusCode。即使请求没有网络错误,状态码也可能表示业务层面的失败,比如404、500等。
常见的做法是与http.StatusOK(即200)进行比较:
- resp, err := http.Get("https://api.example.com/data")
- if err != nil {
- log.Fatal(err)
- }
- defer resp.Body.Close()
- if resp.StatusCode != http.StatusOK {
- log.Printf("请求失败: %d %s", resp.StatusCode, resp.Status)
- return
- }
处理非200状态的场景
不是所有错误都该直接中断。某些接口可能用400表示参数错误,401表示未授权,这些都需要分别处理。
立即学习“go语言免费学习笔记(深入)”;
可以使用switch判断不同状态码,执行相应逻辑:
- switch resp.StatusCode {
- case http.StatusOK:
- // 正常处理
- case http.StatusUnauthorized:
- log.Println("认证失败,请检查Token")
- case http.StatusNotFound:
- log.Println("请求的资源不存在")
- default:
- log.Printf("服务器错误: %d", resp.StatusCode)
- }
确保Body被正确关闭
无论状态码如何,只要resp不为nil,就应调用resp.Body.Close()防止内存泄漏。即使状态是500,也必须读取并关闭Body。
常见错误是在判断状态前就return,忘记关闭:
❌ 错误示例:- if resp.StatusCode != 200 {
- return // 忘记Close,可能导致连接泄露
- }
正确方式是使用defer,或在每个分支中显式关闭。
封装通用错误处理函数
在项目中可封装一个辅助函数,统一处理状态码并返回可读错误:
- func checkResponse(resp *http.Response) error {
- if resp.StatusCode == http.StatusOK {
- return nil
- }
- body, _ := io.ReadAll(resp.Body)
- return fmt.Errorf("HTTP %d: %s, body: %s",
- resp.StatusCode, resp.Status, string(body))
- }
使用时:
- if err := checkResponse(resp); err != nil {
- log.Fatal(err)
- }
基本上就这些。核心是:不要只依赖err判断请求结果,每次都要检查StatusCode,并养成良好习惯关闭Body。这样能显著提升服务的健壮性。










