通八洲科技

如何使用Golang处理HTTP请求错误_GolangHTTP响应错误处理示例

日期:2026-01-02 00:00 / 作者:P粉602998670
HTTP请求失败时resp可能为nil,须先判空再访问;需区分网络层、TLS层、HTTP语义层错误,用errors.As精准判断;StatusCode≥400仍需读取响应体,但须用context和MaxBytesReader限流限超时。

HTTP请求失败时,resp 可能为 nil,必须先判空再读取

Go 的 http.DefaultClient.Do() 在网络不可达、DNS失败、连接超时等情况下会直接返回 err != nil,此时 respnil。如果跳过判空就调用 resp.StatusCoderesp.Body.Close(),会触发 panic。

区分三类错误:网络层、TLS层、HTTP语义层

Go 的 HTTP 错误不是单一类型,不同错误需要不同策略:

errors.As 提取底层错误类型做精准判断

直接用 strings.Contains(err.Error(), "timeout") 不可靠——错误信息可能随 Go 版本变化。推荐用 errors.As 匹配具体错误类型:

var netErr net.Error
if errors.As(err, &netErr) && netErr.Timeout() {
    // 处理超时,例如记录指标或触发重试
}
var urlErr *url.Error
if errors.As(err, &urlErr) && urlErr.Err != nil {
    // 检查 urlErr.Err 是否为 *net.OpError 等
}

响应体读取前必须检查 resp.StatusCode,且用 io.ReadAll 配合 context 控制读取时限

即使 resp.StatusCode >= 400,服务端仍可能返回有意义的错误体(如 {"error": "invalid_token"})。但直接调用 io.ReadAll(resp.Body) 有风险:

ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer cancel()
body, err := io.ReadAll(http.MaxBytesReader(ctx, resp.Body, 1<<20)) // 限制最大 1MB
if err != nil {
    // 处理读取超时或过大响应
}
实际处理中,最易被忽略的是:**把 4xx/5xx 当作网络错误统一重试**,结果导致鉴权失败反复刷 token,或 429 被持续加重。状态码语义必须由业务代码显式分支处理,不能交给通用重试逻辑兜底。