gpt4 book ai didi

go - 使用 errgroup 在第一个错误时取消 goroutine

转载 作者:行者123 更新时间:2023-12-03 10:10:16 26 4
gpt4 key购买 nike

我正在尝试在其中一个遇到错误后取消剩余的 goroutine,或者至少取消其中的 fetch 函数调用。

func fetch(n int, fail bool) (string, error) {
time.Sleep(time.Duration(n) * time.Second)
if fail {
return "", errors.New("an error") //fmt.Errorf("an error")
} else {
return "Hello", nil
}
}

func getData(ctx context.Context, ch chan string, n int, fail bool) error {
fmt.Println("fetch data" + strconv.Itoa(n))

select {
case <-ctx.Done():
return ctx.Err()
default:
if response, err := fetch(n, fail); err != nil {
fmt.Println("error encountered at ", strconv.Itoa(n))
return err
} else {
ch <- response
fmt.Println("fetched data" + strconv.Itoa(n))
return nil
}
}
}

func main() {
res, err := func() (string, error) {
ctx := context.Background()
g, ctx := errgroup.WithContext(ctx)

ch1 := make(chan string, 1)
ch2 := make(chan string, 1)
ch3 := make(chan string, 1)

g.Go(func() error {
defer close(ch1)
return getData(ctx, ch1, 1, true)
})

g.Go(func() error {
defer close(ch2)
return getData(ctx, ch2, 2, false)
})

g.Go(func() error {
defer close(ch3)
return getData(ctx, ch3, 3, false)
})

result := <-ch1 + <-ch2 + <-ch3
return result, g.Wait()
}()

if err != nil {
fmt.Println(err)
} else {
fmt.Println(res)
}
}
输出应如下所示:
fetch data1
fetch data2
fetch data3
error encountered at 1
an error
但实际上是:
fetch data3
fetch data1
fetch data2
error encountered at 1
fetched data2
fetched data3
an error
我理解 errgroup 应该做的是在发现错误时关闭上下文,然后 <-ctx.Done() 应该返回 ctx.Err(),但是这就像上下文从未完成并且 fetch 函数一直在叫。
我认为问题出在 select 语句中。知道我做错了什么或错过了什么吗?
Link to playground

最佳答案

将上下文传递给 fetch 函数并在取消时退出:

func fetch(ctx context.Context, n int, fail bool) (string, error) {  
select {
case <-time.After(time.Duration(n) * time.Second):
if fail {
return "", errors.New("an error")
} else {
return "Hello", nil
}
case <-ctx.Done():
fmt.Println("canceled")
return "CXL", ctx.Err()
}
}
不需要 channel 内容和 GetData。只需将 fetch 的结果分配给变量。
ctx := context.Background()
g, ctx := errgroup.WithContext(ctx)

var r1, r2, r3 string

g.Go(func() error {
var err error
r1, err = fetch(ctx, 1, true)
return err
})

g.Go(func() error {
var err error
r2, err = fetch(ctx, 2, false)
return err
})

g.Go(func() error {
var err error
r3, err = fetch(ctx, 3, false)
return err
})

err := g.Wait()
fmt.Printf("err: %v, 1: %s; 2: %s; 3: %s\n", err, r1, r2, r3)
Run it on the playground .

关于go - 使用 errgroup 在第一个错误时取消 goroutine,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/64627294/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com