问题背景

        在多线程中,每个 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
}