Go语言发送HTTP请求全攻略
时间:2026-05-06 15:43:43 431浏览 收藏
本文深入解析了Go语言中使用net/http发起HTTP请求的核心要点与常见陷阱,强调http.Get虽简洁但必须defer resp.Body.Close()以防连接泄漏,POST应优先选用http.NewRequest配合自定义Client以灵活控制Header、超时和Content-Type,指出仅靠time.After无法真正中断底层TCP连接,必须通过带Timeout的http.Client或精细配置Transport来避免goroutine堆积,同时警示JSON序列化/反序列化时务必检查marshal/unmarshal错误并确保完整读取Body,最后点明超时控制、Body关闭、JSON错误处理和状态码校验这四大坑在真实项目中几乎不可避免——稍有疏忽便可能导致连接耗尽、内存泄漏或静默失败。

用 net/http 发起 GET 请求最简写法
Go 标准库不通过 net 包发 HTTP 请求,而是用 net/http。直接调用 http.Get 是最快上手方式,它内部封装了连接、请求头、状态码处理等细节。
注意:http.Get 返回的 *http.Response 必须手动关闭 Body,否则会持续占用连接和内存。
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.Println(string(body))</p>POST 请求要手动构造 *http.Request
http.Post 虽然存在,但只支持固定 Content-Type(如 application/x-www-form-urlencoded),灵活性差。真正可控的方式是用 http.NewRequest + http.DefaultClient.Do。
http.Post无法设置自定义 Header(比如Authorization)http.NewRequest允许你精确控制 Method、URL、Body、Header、Timeout- 务必检查
resp.StatusCode,http.Do不会因 4xx/5xx 自动报错
req, _ := http.NewRequest("POST", "https://httpbin.org/post", strings.NewReader(`{"name":"go"}`))
req.Header.Set("Content-Type", "application/json")
req.Header.Set("Authorization", "Bearer abc123")
<p>client := &http.Client{Timeout: 10 * time.Second}
resp, err := client.Do(req)
if err != nil {
log.Fatal(err)
}
defer resp.Body.Close()</p><p>if resp.StatusCode != 200 {
log.Printf("unexpected status: %d", resp.StatusCode)
}</p>超时控制必须用 http.Client,不能靠 time.After
仅对 http.Get 或 http.Post 包一层 time.AfterFunc 无法中断底层 TCP 连接,可能造成 goroutine 泄漏。正确做法是创建带 Timeout 的 *http.Client,或更细粒度地设置 Transport 的 DialContext 和 ResponseHeaderTimeout。
Client.Timeout控制整个请求生命周期(DNS + 连接 + 写请求 + 读响应头 + 读响应体)- 若需分别控制连接超时和读响应超时,需自定义
http.Transport - 零值
http.Client{}没有超时,可能无限 hang 住
client := &http.Client{
Timeout: 5 * time.Second,
Transport: &http.Transport{
DialContext: (&net.Dialer{
Timeout: 3 * time.Second,
KeepAlive: 30 * time.Second,
}).DialContext,
ResponseHeaderTimeout: 2 * time.Second,
},
}JSON 请求和响应的典型错误点
用 json.Marshal 构造请求体时,容易忽略错误;用 json.Unmarshal 解析响应时,常忘记先读完 resp.Body。这两步出错都不会触发 HTTP 状态码异常,但会导致静默失败。
json.Marshal失败返回nil字节切片,传给bytes.NewReader后发空体resp.Body是流式读取,多次调用json.NewDecoder(resp.Body).Decode()会报io.EOF- 结构体字段没加
json:tag,导致序列化/反序列化字段为空
data := struct {
Name string `json:"name"`
Age int `json:"age"`
}{"go", 15}
<p>body, err := json.Marshal(data)
if err != nil {
log.Fatal(err) // 必须检查
}</p><p>req, _ := http.NewRequest("POST", url, bytes.NewReader(body))
req.Header.Set("Content-Type", "application/json")</p><p>resp, _ := http.DefaultClient.Do(req)
defer resp.Body.Close()</p><p>var result map[string]interface{}
if err := json.NewDecoder(resp.Body).Decode(&result); err != nil {
log.Fatal(err) // 必须检查
}</p>HTTP 请求看似简单,但超时、Body 关闭、JSON 序列化错误、状态码忽略这四点,在真实项目里几乎必踩。尤其 Body 不关,压测时连接池迅速耗尽。
终于介绍完啦!小伙伴们,这篇关于《Go语言发送HTTP请求全攻略》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!
-
505 收藏
-
503 收藏
-
502 收藏
-
502 收藏
-
502 收藏
-
196 收藏
-
431 收藏
-
497 收藏
-
462 收藏
-
194 收藏
-
232 收藏
-
418 收藏
-
134 收藏
-
307 收藏
-
216 收藏
-
471 收藏
-
424 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习