如何使用 WaitGroup 处理错误和终止 Goroutine

Sev*_*rin 0 concurrency go goroutine

我今天一直在玩 Goroutines、Channels 和 WaitGroup,在阅读了一段时间之后,我终于开始理解这个概念。

我的问题是我不确定在这样工作时如何处理错误,主要是因为我使用了 WaitGroup。使用 WaitGroup 时,我首先添加将要执行的 goroutine 的数量,但是如果在其中一个过程中发生错误怎么办?

package main

import (
    "errors"
    "sync"
)

var waitGroup sync.WaitGroup

func main() {
    c := make(chan int, 10)

    waitGroup.Add(10)

    go doSomething(c)

    waitGroup.Wait()
}

func doSomething(c chan int) {
    for i := 0; i < 10; i++ {
        n, err := someFunctionThatCanError()

        if err != nil {
            // How do I end the routines and WaitGroups here?
        }

        c <- n
        waitGroup.Done()
    }

    close(c)    
}

func someFunctionThatCanError() (int, error) {
    return 1, errors.New("an error")
}
Run Code Online (Sandbox Code Playgroud)

游乐场:https : //play.golang.org/p/ZLsBSqdMD49

我已经尽力提供一个例子来说明我在说什么。一个循环将运行 10 次,doSomething()并且waitGroup.Done()每次迭代都会调用,但是如果在所有这些过程中发生错误someFunctionThatCanError()怎么办,如所示?

当我现在尝试通过返回和/或取消频道来解决它时,我最终陷入僵局,所以我有点不确定从哪里开始。我也不确定如何处理我认为正在等待更多事情发生的 WaitGroup。

任何帮助都非常感谢。

Cer*_*món 5

使用golang.org/x/sync/errgroup等待和处理来自 goroutines 的错误。

package main

import (
    "errors"
    "log"
    "sync"

    "golang.org/x/sync/errgroup"
)

var waitGroup sync.WaitGroup

func main() {
    c := make(chan int, 10)

    var g errgroup.Group

    g.Go(func() error {
        return doSomething(c)
    })

    // g.Wait waits for all goroutines to complete
    // and returns the first non-nil error returned
    // by one of the goroutines.
    if err := g.Wait(); err != nil {
        log.Fatal(err)
    }
}

func doSomething(c chan int) error {
    defer close(c)
    for i := 0; i < 10; i++ {
        n, err := someFunctionThatCanError()
        if err != nil {
            return err
        }
        c <- n
    }
    return nil
}

func someFunctionThatCanError() (int, error) {
    return 1, errors.New("an error")
}
Run Code Online (Sandbox Code Playgroud)

在操场上运行它

  • `sync.WaitGroup` 没有在任何地方使用,但它已定义,是否需要? (4认同)