本文讲解如何将顺序遍历切片并调用函数的操作改为真正的并发执行,避免闭包变量捕获错误和切片越界 panic,使用 sync.waitgroup 安全收集结果。
在 Go 中实现切片元素的并发处理时,核心目标是:对每个元素独立启动 goroutine 执行耗时操作(如 double(i)),不阻塞后续 goroutine 启动,并最终按原始顺序汇总结果。原代码中尝试用 []chan int 存储通道,但存在两个关键错误:
✅ 正确解法无需引入额外通道——直接使用预分配结果切片 + sync.WaitGroup 协调完成信号,简洁、高效且线程安全:
func parallel(arr []int) []int {
outArr := make([]int, len(arr)) // 预分配,保证索引安全
var wg sync.WaitGroup
for i, n
um := range arr {
wg.Add(1)
// 显式传参,避免闭包捕获循环变量
go func(index int, value int) {
defer wg.Done()
outArr[index] = double(value)
}(i, num)
}
wg.Wait() // 主协程阻塞等待所有 goroutine 完成
return outArr
}? 关键要点说明:
⚠️ 注意事项:
综上,Go 的并发编程强调明确的数据所有权与显式的同步机制。相比复杂通道编排,WaitGroup + 预分配切片 是处理“映射型并行任务”(map-reduce 中的 map 阶段)最直观、低开销且不易出错的模式。