HTTP Client 默认不重试,必须手动实现;仅对301/302重定向且限GET/HEAD内部重试,网络层错误(如超时、DNS失败)需业务层封装指数退避重试,并注意Body可重放性。
Go 标准库的 http.Client 在遇到网络错误(如连接超时、DNS失败、TLS握手失败)时**不会自动重试**,哪怕只是临时性抖动。它只在极少数 HTTP 状态码(如 301/302 重定向)下内部重试,且仅限 GET/HEAD 方法。这意味着你调用 client.Do(req) 后拿到 err != nil,基本就是请求彻底失败了——除非你自己加逻辑。
常见错误现象包括:net/http: request canceled (Client.Timeout exceeded while awaiting headers)、dial tcp: i/o timeout、tls: handshake failure。这些都不是服务端返回的 5xx,而是客户端底层连接阶段失败,标准 client 不会重试。
http.Client 自身最轻量的做法是复用原生 http.Client,每次重试都新建 *http.Request 并带上独立的 context.Context 控制单次超时。关键点在于:重试逻辑与请求构造分离,避免共享可变状态。
示例中使用固定 3 次重试、初始 100ms 间隔、最大 1s 上限:
func doWithRetry(client *http.Client, req *http.Request, maxRetries int) (*http.Response, error) {
var resp *http.Response
var err error
baseDelay := 100 * time.Millisecond
for i := 0; i <= maxRetries; i++ {
// 每次重试都新建 context,防止超时继承
ctx, cancel := context.WithTimeout(req.Context(), 5*time.Second)
defer cancel() // 注意:这里 defer 只对最后一次循环生效,实际应移入循环内并及时调用
retryReq := req.Clone(ctx)
resp, err = client.Do(retryReq)
if err == nil {
return resp, nil
}
// 仅对可重试错误才继续,比如网络层错误;4xx 一般不重试
if !isNetworkError(err) {
return nil, err
}
if i < maxRetries {
delay := time.Duration(math.Pow(2, float64(i))) * baseDelay
if delay > time.Second {
delay = time.Second
}
time.Sleep(delay)
}
}
return resp, err}
func isNetworkError(err error) bool {
if urlErr, ok := err.(*url.Error); ok {
return urlErr.Err != nil && (
strings.Contains(urlErr.Err.Error(), "timeout") ||
strings.Contains(urlErr.Err.Error(), "connection refused") ||
strings.Contains(urlErr.Err.Error(), "i/o timeout") ||
strings.Contains(urlErr.Err.Error(), "no such host")
)
}
return false
}
用第三方库 backoff/v4 简化指数退避逻辑
手写退避容易出错(比如忘记限制最大延迟、误用 defer 导致 context 泄漏)。推荐直接用 github.com/cenkalti/backoff/v4,它提供标准化的重试策略和错误分类支持。
注意两点:一是用 backoff.WithContext 包装操作,确保整体超时可控;二是通过 backoff.WithRetryableError 显式定义哪些错误允许重试,避免把 401 或 404 也重试了:
import "github.com/cenkalti/backoff/v4"
func doWithBackoff(client http.Client, req http.Request) (*http.Response, error) {
bo := backoff.WithContext(
backoff.NewExponentialBackOff(),
req.Context(),
)
var resp *http.Response
err := backoff.Retry(func() error {
r, e := client.Do(req.Clone(req.Context()))
if e != nil {
if isNetworkError(e) {
return e // 触发重试
}
return backoff.Permanent(e) // 终止重试
}
resp = r
// 可选:对 5xx 响应也视为可重试
if resp.StatusCode >= 500 && resp.StatusCode < 600 {
return fmt.Errorf("server error: %d", resp.StatusCode)
}
return nil
}, bo)
return resp, err}
重试时别忽略 Request.Body 的可重放性
这是最容易踩的坑:如果原始 req.Body 是 bytes.Reader 或 strings.Reader,那可以反复读;但如果是 os.File、net.Conn 或已关闭的 io.ReadCloser,重试时再调用 req.Body.Read() 就会返回 io.EOF 或 panic。
解决方案只有两个:
bytes.Buffer),再用 bytes.NewReader(buf.Bytes()) 构造新 Body —— 适合小数据(bytes.NewReader(payload) —— 更安全,但需业务层持有 payload如果你用 json.Marshal 构造 POST body,那就天然可重放;但若直接传 os.Stdin 或临时文件句柄,重试必然失败。
# js
# git
# json
# go
# github
# golang
# ai
# dns
# 状态码
# 标准库
# EOF
# if
# while
# 封装
# Error
# bool
# nil
# delete
# http
# 重试
# 重放
# 仅对
# 重定向
# 这是
# 就会
# 那就
# 句柄
# 一是
# 你自己
相关文章:
建站之星CMS五站合一模板配置与SEO优化指南
相亲简历制作网站推荐大全,新相亲大会主持人小萍萍资料?
官网自助建站系统:SEO优化+多语言支持,快速搭建专业网站
建站org新手必看:2024最新搭建流程与模板选择技巧
实惠建站价格推荐:2025年高性价比自助建站套餐解析
教学网站制作软件,学习*后期制作的网站有哪些?
如何快速查询网址的建站时间与历史轨迹?
Python lxml的etree和ElementTree有什么区别
c++怎么编写动态链接库dll_c++ __declspec(dllexport)导出与调用【方法】
建站之星好吗?新手能否轻松上手建站?
如何用免费手机建站系统零基础打造专业网站?
如何破解联通资金短缺导致的基站建设难题?
微信推文制作网站有哪些,怎么做微信推文,急?
已有域名如何免费搭建网站?
Swift中循环语句中的转移语句 break 和 continue
定制建站流程解析:需求评估与SEO优化功能开发指南
广平建站公司哪家专业可靠?如何选择?
零服务器AI建站解决方案:快速部署与云端平台低成本实践
如何在香港服务器上快速搭建免备案网站?
专业制作网站的公司哪家好,建立一个公司网站的费用.有哪些部分,分别要多少钱?
c# F# 的 MailboxProcessor 和 C# 的 Actor 模型
宝塔建站助手安装配置与建站模板使用全流程解析
洛阳网站制作公司有哪些,洛阳的招聘网站都有哪些?
定制建站方案优化指南:企业官网开发与建站费用解析
建站之星如何快速生成多端适配网站?
购物网站制作费用多少,开办网上购物网站,需要办理哪些手续?
PHP正则匹配日期和时间(时间戳转换)的实例代码
建站主机选哪种环境更利于SEO优化?
建站之星免费模板:自助建站系统与智能响应式一键生成
宝塔新建站点为何无法访问?如何排查?
一键网站制作软件,义乌购一件代发流程?
建站之星Pro快速搭建教程:模板选择与功能配置指南
企业在线网站设计制作流程,想建设一个属于自己的企业网站,该如何去做?
建站主机助手选型指南:2025年热门推荐与高效部署技巧
Avalonia如何实现跨窗口通信 Avalonia窗口间数据传递
清除minerd进程的简单方法
如何快速查询网站的真实建站时间?
如何快速生成高效建站系统源代码?
如何通过IIS搭建网站并配置访问权限?
建站之星代理如何获取技术支持?
香港服务器建站指南:外贸独立站搭建与跨境电商配置流程
北京网站制作的公司有哪些,北京白云观官方网站?
如何用AWS免费套餐快速搭建高效网站?
电商网站制作价格怎么算,网上拍卖流程以及规则?
C#如何序列化对象为XML XmlSerializer用法
如何选择高性价比服务器搭建个人网站?
制作网站的软件免费下载,免费制作app哪个平台好?
设计网站制作公司有哪些,制作网页教程?
如何配置WinSCP新建站点的密钥验证步骤?
Java解压缩zip - 解压缩多个文件或文件夹实例
*请认真填写需求信息,我们会在24小时内与您取得联系。