Go中异步HTTP请求本质是用goroutine并发执行多个请求以避免串行等待;通过goroutine+channel配合net/http实现非阻塞效果,为每个请求启goroutine并用channel收集结果。

在 Go 中实现异步 HTTP 请求,本质是利用 goroutine 并发执行多个请求,避免串行等待、提升整体响应速度。Go 本身没有“异步 I/O”概念(如 Node.js 的 event loop),但通过轻量级 goroutine + channel 配合标准 net/http,就能高效达成非阻塞效果。
用 goroutine 并发发起多个请求
最直接的方式:为每个请求启动一个 goroutine,用 channel 收集结果。适合请求数量固定、逻辑简单的情况。
- 定义结果结构体,包含响应数据、错误、原始 URL 等上下文信息
- 启动 goroutine 执行
http.Get(或自定义 client.Do),完成后将结果写入 channel - 主 goroutine 用
for range从 channel 接收所有结果(注意设置超时或限制并发数)
示例片段:
go func(url string) {resp, err := http.Get(url)
resultChan }(url)
控制并发数,防止资源耗尽
不加限制地启动数百个 goroutine 可能打垮客户端(文件描述符不足)或服务端(被限流)。推荐用带缓冲的 channel 或 semaphore 模式限流。
立即学习“go语言免费学习笔记(深入)”;
- 创建容量为 N 的 channel 作为信号量:
sem := make(chan struct{}, 10) - 每个请求前先写入:
sem ;完成后读出: - 也可使用第三方库如
golang.org/x/sync/semaphore更规范
使用 context 控制超时与取消
单个请求卡住会拖慢整个流程。务必为每个请求绑定 context.Context,设定超时或可主动取消。
- 用
context.WithTimeout(ctx, 5*time.Second)包装请求上下文 - 传给
http.NewRequestWithContext,再由 client 发起 - 若上游调用提前结束(如用户关闭页面),context.Done() 触发,HTTP 客户端自动中断连接
组合使用:并发 + 限流 + 超时 + 错误聚合
生产环境建议封装成可复用函数,返回统一结果切片,并区分成功/失败项。
- 输入:URL 列表、最大并发数、全局超时时间
- 内部用 worker pool 模式:固定数量 goroutine 从任务 channel 拿 URL 执行
- 结果汇总到 slice,按需排序或过滤;失败请求可记录日志或重试一次
- 避免用
time.Sleep等待,始终用select+ctx.Done()做安全退出
不复杂但容易忽略细节,关键是把 goroutine 当作“并发单元”,而不是“异步回调”。










