如何在Golang中使用net/http发送请求_Golang net/http客户端方法

13次阅读

应使用自定义 http.Client 替代 http.Get:可设超时、Header、重试,避免连接泄漏;发 jsON POST 需用 http.NewRequest + client.Do 并显式设 Content-Type;务必关闭或读取 resp.Body。

如何在Golang中使用net/http发送请求_Golang net/http客户端方法

如何用 http.Get 发起最简 GET 请求

直接调用 http.Get 是最快捷的方式,但它会使用默认的 http.DefaultClient,不支持超时、自定义 Header 或重试控制。

  • 必须手动关闭响应体(resp.Body.Close()),否则会泄漏 HTTP 连接
  • 无法设置请求超时 —— 默认阻塞直到 TCP 连接建立 + 响应返回,容易卡死
  • 若服务端返回非 2xx 状态码http.Get 不报错,需手动检查 resp.StatusCode
resp, err := http.Get("https://httpbin.org/get") if err != nil {     log.Fatal(err) } defer resp.Body.Close() // 必须加 

body, _ := io.ReadAll(resp.Body) fmt.Printf("status: %d, body: %s", resp.StatusCode, string(body))

为什么应该用 http.Client 替代默认客户端

http.Client 是可配置的核心类型,所有高级控制都依赖它。默认客户端只是 &http.Client{} 的实例,没有设任何超时。

  • Timeout 字段控制整个请求生命周期(连接 + 写入 + 读取)
  • Transport 可定制连接池、TLS 配置、代理、重试逻辑
  • 复用 http.Client 实例是安全且推荐的 —— 它是并发安全的
client := &http.Client{     Timeout: 5 * time.Second, } resp, err := client.Get("https://httpbin.org/delay/3")

如何发送带 Header 和 json Body 的 POST 请求

gohttp.Post 函数看似方便,但只支持固定 Content-Type(application/x-www-form-urlencoded 或纯文本),发 JSON 必须用 http.NewRequest + client.Do

  • JSON 数据要先序列化为 []byte,再传给 bytes.NewReader
  • 必须显式设置 Content-Type: application/json,否则服务端可能解析失败
  • 不要用 http.Post 发 JSON —— 它会忽略你传的 Header
data := map[string]string{"name": "alice"} jsonBytes, _ := json.Marshal(data) 

req, _ := http.NewRequest("POST", "https://www.php.cn/link/dc076eb055ef5f8a60a41b6195e9f329", bytes.NewReader(jsonBytes)) req.Header.Set("Content-Type", "application/json") req.Header.Set("Authorization", "Bearer xyz")

client := &http.Client{Timeout: 10 * time.Second} resp, err := client.Do(req)

常见错误:未读取或未关闭 resp.Body 导致连接泄漏

只要调用了 client.Dohttp.Get,就一定有 resp.Body。即使你只关心状态码,也必须读完或关闭它,否则底层连接不会归还给连接池。

立即学习go语言免费学习笔记(深入)”;

  • io.copy(io.Discard, resp.Body) 忽略响应体内容
  • io.ReadAll 读取全部后仍要 Close()
  • defer 中关,但注意:如果 resp 是 nil(比如请求失败),defer resp.Body.Close() 会 panic
resp, err := client.Do(req) if err != nil {     log.Fatal(err) } defer resp.Body.Close() // 此时 resp 不为 nil,安全 

// 如果只想检查状态码,又不想读 body: , = io.Copy(io.Discard, resp.Body) // 必须执行这一行

HTTP 客户端行为高度依赖底层 http.Transport,而它的默认配置(如 MaxIdleConnsPerHost: 2)在高并发场景下极易成为瓶颈 —— 这个细节多数人写完第一个 http.Client 就忘了调。

text=ZqhQzanResources