Go 中实现 HTTP 请求重试需控制次数、指数退避延迟、错误类型判断(仅重试网络错误或 5xx),并解耦重试逻辑与 HTTP 调用,推荐用函数选项封装 DoWithRetry。

在 Go 中实现 HTTP 请求重试,关键不是简单地循环调用 http.Do,而是要有可控的重试次数、可配置的延迟策略(如指数退避)、错误类型判断(比如只重试网络错误或 5xx,跳过 4xx),以及避免阻塞主逻辑。下面用自定义函数 + 回退机制(backoff)来实现一个实用、可复用的重试方案。
使用指数退避(exponential backoff)是最常见的回退机制:每次失败后等待时间翻倍(加上随机抖动防雪崩)。Go 标准库没有直接提供,但可以用 time.Sleep 自行实现:
delay = min(initial * (multiplier ^ attempt), maxDelay)
不是所有错误都该重试。例如:
net.OpError)、超时(context.DeadlineExceeded)、服务端临时错误(HTTP 500/502/503/504)shouldRetry(err error, resp *http.Response) bool 函数做统一判断把重试逻辑和 HTTP 调用解耦,推荐用函数选项(functional options)或结构体配置方式。示例核心逻辑:
立即学习“go语言免费学习笔记(深入)”;
func DoWithRetry(client *http.Client, req *http.Request, opts ...RetryOption) (*http.Response, error) {
o := applyOptions(opts...) // 合并配置
var lastErr error
for i := 0; i <= o.maxRetries; i++ {
resp, err := client.Do(req)
if err == nil && isRetryableStatusCode(resp.StatusCode) {
err = fmt.Errorf("HTTP %d", resp.StatusCode)
}
if err == nil {
return resp, nil
}
lastErr = err
if i == o.maxRetries || !shouldRetry(err, nil) {
break
}
delay := calculateBackoff(i, o.initialDelay, o.multiplier, o.maxDelay)
time.Sleep(delay)
// 注意:若需重试 POST/PUT 等非幂等请求,应确保 req.Body 可重放(如用 bytes.NewReader)
}
return nil, lastErr
}默认 http.Request.Body 是单次读取的,重试时会因 body 已关闭或 EOF 导致失败:
bytes.Buffer 或 []byte),每次重试重建 req.Body = io.NopCloser(bytes.NewReader(data))
context.Context 传入 http.Client,并在重试中保留 cancel/timeout 控制力基本上就这些。不复杂但容易忽略细节——尤其是错误分类和 body 重放。用好指数退避 + 精准重试判断,就能让 HTTP 客户端稳得多。
以上就是如何在Golang中处理HTTP请求重试_使用自定义函数和回退机制的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号