登录
首页 >  Golang >  Go问答

主协程在这种情况下为什么会被阻塞,导致了死锁的发生?

来源:stackoverflow

时间:2024-02-06 12:16:12 313浏览 收藏

有志者,事竟成!如果你在学习Golang,那么本文《主协程在这种情况下为什么会被阻塞,导致了死锁的发生?》,就很适合你!文章讲解的知识点主要包括,若是你对本文感兴趣,或者是想搞懂其中某个知识点,就请你继续往下看吧~

问题内容

package main

import "fmt"

func square(numbers chan int, squares chan int) {
    for n := range numbers {
        squares <- n * n
    }
    close(squares)
}

func main() {
    numbers := make(chan int)
    squares := make(chan int)

    go square(numbers, squares)

    for i := 0; i < 10; i++ {
        numbers <- i
    }
    close(numbers)

    for s := range squares {
        fmt.Println(s)
    }
}

我的意思是,我知道要使此代码正常工作,应该将数字发送到单独的 goroutine 中的 numbers 通道,例如:

go func() {
for i := 0; i < 10; i++ {
    numbers <- i
}

}

话虽如此,我发现很难解释为什么会出现僵局。我很清楚调度程序不能保证执行顺序。但是,在循环中第一次发送到 numbers 通道时,主 goroutine 被阻塞,但随后调度程序可能会开始执行 square goroutine,然后它们会来回通信,这不是这样吗?


正确答案


主 goroutine 被阻塞的原因是,在这种情况下,将数据发送到 squares 通道后,您没有从 squares 通道读取任何值。

当你执行 numbers <- i 时,你的 go square goroutine 将接收该值并将其发送到 squares 通道。但是,与此同时,您的主 Goroutine 不会从 sqaures 通道接收值,因为您的主 Goroutine 仍然将数据发送到 Numbers 通道。

这意味着你的主协程永远不会执行这一行 for s := range squares ,然后它会导致死锁。

为了正确运行此代码,您可以将其修改为如下所示。

package main

import "fmt"

func square(numbers chan int, squares chan int) {
    for n := range numbers {
        squares <- n * n
    }
    close(squares)

}

func main() {
    numbers := make(chan int)
    squares := make(chan int)

    go square(numbers, squares)

    go func() {
        for i := 0; i < 10; i++ {
            numbers <- i
        }
        close(numbers)
    }()

    for s := range squares {
        fmt.Println(s)
    }
}

到这里,我们也就讲完了《主协程在这种情况下为什么会被阻塞,导致了死锁的发生?》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>