问题背景
在多线程中,每个 goruntine 中可能共享同一个 error 变量,此时当发生 error 事件后当对应 goruntine协程退出后其他协程依然在执行中,因此很有可能会再次修改 error 变量进行覆盖。
解决方案
使用带有缓冲 channel 通道,在发生 error 时将 error 数据写入管道中,这样在所有多线程执行完任务后就可以捕捉到所有发生的 error 错误信息。
此方案需要等待所有多线程执行结束退出(wg.Wait() 阻塞结束),因此适用的场景是需要捕捉多线程中所有发生 error 的信息。如果在第一次发生 error 后就结束返回可以使用阻塞通道来实现~
代码如下
var err error
var wg sync.WaitGroup
ids := []int64{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
wg.Add(len(ids))
errCh := make(chan error, len(ids))
for _, v := range ids {
id := v
go func() {
defer func() {
if err != nil {
errCh <- err
}
wg.Done()
}()
err = Func()
if err != nil {
return
}
err = Func2()
if err != nil {
return
}
err = Func3()
if err != nil {
return
}
}()
}
wg.Wait()
close(errCh)
for chErr := range errCh {
return chErr
}