http.Get是最简GET请求方式,但无超时和状态码校验;生产环境应使用自定义http.Client设置超时、复用连接,并用url.Values安全拼接参数;需手动检查StatusCode及解析响应体判断业务错误。

用 http.Get 发起最简 GET 请求
Go 标准库的 http.Get 是最直接的方式,适合无额外头、无超时控制的简单场景。它本质是调用 http.DefaultClient.Get,底层复用连接池和默认配置。
但要注意:它不设超时,一旦后端卡住或网络异常,协程会永久阻塞;也不校验 HTTP 状态码,200 和 500 都返回 nil 错误。
resp, err := http.Get("https://httpbin.org/get")
if err != nil {
log.Fatal(err)
}
defer resp.Body.Close()
body, _ := io.ReadAll(resp.Body)
fmt.Println(string(body))
用 http.Client 控制超时与重试
生产环境必须显式创建 http.Client 实例,否则无法设置超时、无法复用连接、也无法注入自定义 Transport。
-
Timeout作用于整个请求生命周期(DNS + 连接 + 写请求 + 读响应) - 若需更细粒度控制(如单独设连接超时),得配置
http.Transport - Go 默认启用 HTTP/1.1 连接复用,
http.Client实例应复用,不要每次请求都新建
client := &http.Client{
Timeout: 5 * time.Second,
}
resp, err := client.Get("https://httpbin.org/get")
// ... 处理 resp 和 err带查询参数的 GET:用 url.Values 拼接
手动拼接 URL 容易出错(编码缺失、& 位置错、空值处理不当)。Go 推荐用 url.Values 构建查询字符串,再注入到 *url.URL 中。
注意:url.Parse 后要检查错误;url.Values.Encode() 自动做 URL 编码;不要把参数硬写进字符串里。
baseURL, _ := url.Parse("https://httpbin.org/get")
params := url.Values{}
params.Set("q", "go http")
params.Set("page", "1")
baseURL.RawQuery = params.Encode()
resp, _ := http.Get(baseURL.String())
处理非 200 响应和重定向
http.Client 默认自动跟随 301/302 重定向(最多 10 次),可通过设置 CheckRedirect 函数干预。而服务端返回 4xx 或 5xx 时,resp.StatusCode 仍为对应状态码,err 为 nil —— 这点和 Python 的 requests 不同,容易漏判错误响应。
- 检查
resp.StatusCode 是常见做法 - 若需禁用重定向,把
CheckRedirect设为func(req *http.Request, via []*http.Request) error { return http.ErrUseLastResponse } - 重定向过程中,原始请求头不会自动复制到跳转后的请求,如有需要得手动处理
真正难处理的是中间件透传失败、上游返回 200 但 body 里是错误 JSON —— 这类必须靠业务层解析响应体判断,HTTP 客户端无能为力。









