在 Go 中实现网络请求超时与重试需组合 context.Context 与 http.Client:用 context.WithTimeout 控制全链路超时并及时 cancel;Client 级超时仅作兜底;重试应基于错误类型判断、指数退避、新建 request 和 context,并封装为可复用接口。
在 Go 中实现网络请求的超时控制和重试逻辑,核心是组合 context.Context 与 http.Client 的超时配置,并封装可重试的请求逻辑。不依赖第三方库也能做到简洁可靠。
HTTP 客户端本身支持超时,但更推荐用 context.WithTimeout 统一管理——它能中断阻塞的 DNS 解析、连接建立、TLS 握手和响应读取全过程。
ctx,
cancel := context.WithTimeout(context.Background(), 5*time.Second)
http.NewRequestWithContext(ctx, ...)
cancel() 避免 goroutine 泄漏(即使请求已结束)context.DeadlineExceeded 可区分超时和其他失败Client 的 Timeout 字段仅作用于整个请求生命周期(从发起到收到完整响应),无法中断中间阶段;而 Transport 的各字段(如 DialContextTimeout)可精细控制底层行为。
Transport.DialContext 超时(连接建立)和 ResponseHeaderTimeout(等待响应头)client := &http.Client{Timeout: 10 * time.Second} 是兜底,不能代替 context重试不是无脑循环,需满足:可重试条件判断、指数退避、最大重试次数限制、上下文取消传播。
context.DeadlineExceeded(说明上次超时可能只是抖动)time.Sleep(backoff) 实现退避,初始 100ms,每次 ×2,上限 1s(避免雪崩)把上述逻辑收拢成一个函数或结构体,对外暴露 clean 接口:
不复杂但容易忽略细节,关键是 context 生命周期、错误分类和退避节奏。