Go 中汇总并发任务结果需协调 goroutine 生命周期并安全收集返回值:channel 适合传结果或处理不确定任务,WaitGroup 适合仅需“全部完成”的轻量场景,二者可组合;用带缓冲 channel(容量为 N)收集确定数量结果,注意防泄漏和死锁。
在 Go 中汇总并发任务的结果,核心是协调 goroutine 的生命周期并安全收集返回值。channel 和 sync.WaitGroup 各有适用场景:channel 更适合需要传递结果、控制流或处理不确定数量/耗时的任务;WaitGroup 更轻量,适合只关心“全部完成”而不需中间结果的场景。两者也可组合使用。
当启动 N 个 goroutine 并期望收到 N 个结果时,可创建带缓冲的 channel(容量为 N),每个 goroutine 发送结果后关闭 channel 或由主协程接收完即止。注意避免 goroutine 泄漏
和死锁。
results := make(chan int, numTasks)
results (无需 close)
for i := 0; i (推荐,不依赖 close)
for range results,除非明确会在所有 sender 结束后 close(results)
当任务可能失败、耗时差异大,或需提前终止时,建议用结构体封装结果,并配合 select + 超时或取消控制。
type Result struct { Value int; Err error }
select 处理成功、错误、超时:select { case r := context.Context 可实现更灵活的取消传播(如某个任务失败就中止其余)如果只需等待完成、结果可写入预先分配的切片或 map,且无并发写冲突风险,WaitGroup 更简洁。关键是要保证对共享数据的访问是线程安全的。
var wg sync.WaitGroup 和线程安全容器(如 sync.Map)或加锁的切片defer wg.Done(),并在临界区写入结果(例如用 mu.Lock()/Unlock())wg.Wait() 后再读取汇总数据实际项目中常将两者结合:WaitGroup 管理 goroutine 生命周期,channel 传递结果,兼顾安全性与表达力。
wg.Add(1),内部 defer wg.Done()
wg.Wait() 后仍可从 channel 接收(若未关闭,需配合计数)close(channel),然后 range 安全遍历