登录
首页 >  Golang >  Go教程

Go语言发送HTTP请求方法详解

时间:2026-04-10 15:54:45 300浏览 收藏

本文深入剖析了Go语言中HTTP请求的最佳实践,明确指出`http.Get`仅适用于本地调试,而生产环境必须自建配置完善的`*http.Client`——需显式设置超时、定制`Transport`以管控连接池,并严格遵循`defer resp.Body.Close()`防止连接泄漏;同时强调JSON POST应使用`bytes.NewBuffer`避免重复序列化和内存逃逸,且`http.Client`必须全局复用以充分利用连接复用、DNS缓存与TLS会话,大幅提升高并发下的稳定性和性能。

Go语言怎么发HTTP请求_Go语言net/http客户端教程【进阶】

怎么用 http.Get 发最简 GET 请求

直接调用 http.Get 最快,但别忘了它不自动处理重定向、不设超时、不复用连接——线上服务一压就挂。http.Get 本质是用默认的 http.DefaultClient,而这个 client 的 Timeout 是 0(无限等待),Transport 也没限制连接池大小。

  • 只适合调试或内部脚本,比如本地查个健康接口:http.Get("http://localhost:8080/health")
  • 真实场景必须自建 *http.Client,显式设 TimeoutTransport
  • 别用 http.Get 接第三方 API,DNS 解析卡住、后端响应慢、TCP 握手失败都会导致 goroutine 永久阻塞

为什么 resp.Body 必须手动 Close

Go 的 HTTP client 不会自动关闭响应体,漏掉 resp.Body.Close() 会导致底层 TCP 连接无法归还给连接池,连接数越积越多,最终触发 net/http: request canceled (Client.Timeout exceeded while awaiting headers)dial tcp: lookup failed

  • 必须在 defer resp.Body.Close() 前检查 err 是否非 nil,否则 panic:if err != nil { return err }; defer resp.Body.Close()
  • 如果用 ioutil.ReadAll(Go 1.16+ 改为 io.ReadAll),也要在读完后关;流式读取(如 io.Copy)同理
  • 哪怕只读 resp.StatusCode,也得关——body 可能还有未读数据,不关就锁死连接

POST JSON 数据该用 json.Marshal 还是 bytes.NewBuffer

bytes.NewBuffer 包一层 json.Marshal 输出的字节切片,不是为了“更标准”,而是避免重复序列化和内存逃逸。直接传 bytes.NewBuffer(jsonBytes)http.NewRequest,比每次请求都 json.Marshal + 新建字符串再转 []byte 更省。

  • 正确写法:data, _ := json.Marshal(req); req, _ := http.NewRequest("POST", url, bytes.NewBuffer(data))
  • 错误写法:http.NewRequest("POST", url, strings.NewReader(string(data))) —— 多一次 string 转换,触发额外内存分配
  • 注意设置 header:req.Header.Set("Content-Type", "application/json"),否则服务端可能解析失败

并发发请求时 http.Client 能复用吗

能,而且必须复用。每个 *http.Client 自带连接池(通过 Transport),复用 client 才能复用 TCP 连接、复用 DNS 缓存、复用 TLS 会话。新建 client 相当于扔掉整个连接池,高并发下瞬间打满文件描述符。

  • 全局声明一个 client:var httpClient = &http.Client{Timeout: 10 * time.Second}
  • 需要定制 Transport 时,不要改 DefaultTransport,而是 new 一个:&http.Transport{MaxIdleConns: 100, MaxIdleConnsPerHost: 100}
  • 别在 handler 里 new client,尤其别用 context.WithTimeout 包 client——timeout 应该由 client 自己控制,不是靠 context cancel 间接中断

真正难的是超时粒度:你没法对单个请求设不同 timeout,除非为每种场景配独立 client。多数时候,按业务分组(如“支付回调”“日志上报”)设不同 client 更可控。

理论要掌握,实操不能落!以上关于《Go语言发送HTTP请求方法详解》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>