为什么不执行延迟功能?

时间:2020-03-21 14:23:37

标签: for-loop go concurrency channel deferred

我正在努力理解Go中的并发性。

package main

import "fmt"

func sendValues(myIntChannel chan int) {
    for i := 0; i < 5; i++ {
        myIntChannel <- i //sending value
    }
}

func main() {
    myIntChannel := make(chan int)
    defer close(myIntChannel)
    go sendValues(myIntChannel)

    for value := range myIntChannel {
        fmt.Println(value) //receiving value
    }
}

上面的代码给出以下输出:

0
1
2
3
4
fatal error: all goroutines are asleep - deadlock!

goroutine 1 [chan receive]:
main.main()
    /Users/spikki/Desktop/GoLearning/go_channel.go:51 +0x10b

据我了解,延迟功能将在其周围的功能完成后执行。我无法解释。

如果我使用for循环从通道接收值,其工作方式如下。

for i := 0; i < 5; i++ {
    fmt.Println(<-myIntChannel) //receiving value
}

有人可以帮我理解这个概念吗?

2 个答案:

答案 0 :(得分:3)

for ... range上的通道只有在已从通道接收到所有值且通道已关闭后才终止。

在您的示例中,您希望使用延迟功能关闭通道,但这仅在main()返回时运行。但是main()仅在循环结束时才会返回。这是导致死锁的原因。 for循环等待通道关闭,关闭通道等待for循环结束。

当您使用循环从通道接收正好5个值时,它起作用了,因为启动的goroutine在其上发送了5个值。此循环不等待通道关闭,因此循环可以结束,main()函数也可以结束。

这就是为什么发件人应该关闭通道(而不是接收者),并立即解决问题的原因:

func sendValues(myIntChannel chan int) {
    for i := 0; i < 5; i++ {
        myIntChannel <- i //sending value
    }
    close(myIntChannel)
}

func main() {
    myIntChannel := make(chan int)
    go sendValues(myIntChannel)

    for value := range myIntChannel {
        fmt.Println(value) //receiving value
    }
}

输出(在Go Playground上尝试):

0
1
2
3
4

答案 1 :(得分:1)

要用一些不同的术语来解释它,您的代码将执行以下操作:

func main() {
    ...
    while myIntChannel is not closed {
        ...
    }
    close myIntChannel 
}

现在您可以看到死锁的来源。

虽然以上答案有效,但如果您更喜欢使用defer,也可以尝试以下方法:

func sendValues(myIntChannel chan int) {
    defer close(myIntChannel)
    for i := 0; i < 5; i++ {
        myIntChannel <- i //sending value
    }
}

func main() {
    myIntChannel := make(chan int)

    go sendValues(myIntChannel)

    for value := range myIntChannel {
        fmt.Println(value) //receiving value
    }
}