登录
首页 >  Golang >  Go教程

Golangnet/http请求发送全解析

时间:2025-12-31 19:32:38 287浏览 收藏

学习Golang要努力,但是不要急!今天的这篇文章《Golang net/http发送请求方法详解》将会介绍到等等知识点,如果你想深入学习Golang,可以关注我!我会持续更新相关文章的,希望对大家都能有所帮助!

应使用自定义 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() // 必须加
<p>body, _ := io.ReadAll(resp.Body)
fmt.Printf("status: %d, body: %s", resp.StatusCode, string(body))</p>

为什么应该用 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 请求

Go 的 http.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)
<p>req, _ := http.NewRequest("POST", "<a target='_blank'  href='https://www.17golang.com/gourl/?redirect=MDAwMDAwMDAwML57hpSHp6VpkrqbYLx2eayza4KafaOkbLS3zqSBrJvPsa5_0Ia6sWuR4Juaq6t9nq5roGCUgXuytMyerpV6iZXHe3vUmsyZr5vTk6bFeWmuyXyFmnmyhqK_qrtog3Z4lb6InJSSp62xhph6mq-cm2i0jaCcfbOdorLdtKSCiYSXva6coQ' rel='nofollow'>https://httpbin.org/post</a>", bytes.NewReader(jsonBytes))
req.Header.Set("Content-Type", "application/json")
req.Header.Set("Authorization", "Bearer xyz")</p><p>client := &http.Client{Timeout: 10 * time.Second}
resp, err := client.Do(req)</p>

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

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

  • 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,安全
<p>// 如果只想检查状态码,又不想读 body:
<em>, </em> = io.Copy(io.Discard, resp.Body) // 必须执行这一行</p>

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

文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《Golangnet/http请求发送全解析》文章吧,也可关注golang学习网公众号了解相关技术文章。

前往漫画官网入口并下载 ➜
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>