Go 关闭通道的正确方法

Go 关闭通道的正确方法,go,goroutine,Go,Goroutine,我想要一组goroutine,它们将从许多服务器获取一些信息。我正在简化下面的代码,使其更具可读性。它似乎工作得很好,但在所有任务完成后,它会惊慌失措,因为我从未关闭频道。问题是我不确定我应该在哪里关闭它 我需要你的帮助: 告诉我应该在代码中的什么地方关闭频道 告诉我这段代码的整体逻辑在习惯用法上是否正确 我的代码 当一个频道上只有一个发送者时,该发送者通常负责在必要时关闭该频道。请记住,虽然清理不需要关闭通道,但只有在需要发出通道已关闭的信号时才需要关闭通道 当有多个发送者时,您需要与所有

我想要一组goroutine,它们将从许多服务器获取一些信息。我正在简化下面的代码,使其更具可读性。它似乎工作得很好,但在所有任务完成后,它会惊慌失措,因为我从未关闭频道。问题是我不确定我应该在哪里关闭它

我需要你的帮助:

  • 告诉我应该在代码中的什么地方关闭频道
  • 告诉我这段代码的整体逻辑在习惯用法上是否正确
我的代码
当一个频道上只有一个发送者时,该发送者通常负责在必要时关闭该频道。请记住,虽然清理不需要关闭通道,但只有在需要发出通道已关闭的信号时才需要关闭通道

当有多个发送者时,您需要与所有发送者协调完成,这可以通过

ch:=make(chan字符串)
var wg sync.WaitGroup
对于i:=0;i<10;i++{
工作组.添加(1)

go func(c chan当通道上只有一个发送方时,该发送方通常会在必要时负责关闭通道。请记住,虽然清理不需要关闭通道,但仅当您需要发出关闭通道的信号时才需要关闭通道

当有多个发送者时,您需要与所有发送者协调完成,这可以通过

ch:=make(chan字符串)
var wg sync.WaitGroup
对于i:=0;i<10;i++{
工作组.添加(1)

go func(c)非常感谢你的回答。我不知道当涉及频道时我需要一个等待组。一个问题,为什么工作组等待并关闭(ch)需要在goroutine中吗?我尝试删除goroutine,但它被锁定。您不需要WaitGroup,因为您使用了通道,它是管理goroutine生命周期的有用工具。您可以手动计算结果:。
wg.Wait()
需要进入goroutine本身,因为在阅读允许goroutine完成的频道之前,你不能阻止等待goroutine完成。哦,我现在知道了。非常感谢!非常感谢你的回答。我不知道当涉及频道时我需要一个waitGroup。一个问题,为什么wg.wait和close(ch)本身需要在goroutines中吗?我试图删除goroutine,但它被锁定。您不需要WaitGroup,因为您使用了通道,它是管理goroutines生命周期的有用工具。您可以手动计算结果:。
wg.Wait()
需要在goroutine中,因为在阅读允许goroutine完成的频道之前,您无法阻止等待goroutine完成。哦,我现在就知道了。非常感谢!
func main() {
        ch := make(chan string)


        for i:= 0; i < 10 ; i++ {
                go func(c chan <- string,t int){
                        time.Sleep( time.Duration(rand.Intn(3000)) * time.Millisecond )
                        c <- strconv.Itoa(t) + " : Done " + strconv.Itoa(rand.Intn(3000))
                }(ch,i)
        }
        for val := range ch {
                fmt.Println(val)

        }
}
$ go run test_channels.go
0 : Done 1694
6 : Done 511
3 : Done 162
2 : Done 89
8 : Done 2728
5 : Done 1274
1 : Done 2211
9 : Done 1445
4 : Done 2237
7 : Done 1106
fatal error: all goroutines are asleep - deadlock!

goroutine 1 [chan receive]:
main.main()
        /home/matias/projects/src/github.com/matias/test/test_channels.go:22 +0x138
exit status 2
ch := make(chan string)
var wg sync.WaitGroup

for i := 0; i < 10; i++ {
    wg.Add(1)
    go func(c chan<- string, t int) {
        defer wg.Done()
        time.Sleep(time.Duration(rand.Intn(3000)) * time.Millisecond)
        c <- strconv.Itoa(t) + " : Done " + strconv.Itoa(rand.Intn(3000))
    }(ch, i)
}

go func() {
    wg.Wait()
    close(ch)
}()