登录
首页 >  Golang >  Go问答

比较 Goroutine 中的"go"关键字和"without"

来源:stackoverflow

时间:2024-03-02 17:12:26 452浏览 收藏

最近发现不少小伙伴都对Golang很感兴趣,所以今天继续给大家介绍Golang相关的知识,本文《比较 Goroutine 中的"go"关键字和"without"》主要内容涉及到等等知识点,希望能帮到你!当然如果阅读本文时存在不同想法,可以在评论中表达,但是请勿使用过激的措辞~

问题内容

以下代码记录了错误:

致命错误:所有 goroutine 都在睡觉 - 死锁!

package main

import "fmt"

func main() {
    ch := make(chan int)
    ch <- 1
    fmt.println(<-ch)
}

但是当我将代码更改为:

package main

import "fmt"

func assign (ch chan int) {
    ch <- 1
}

func main() {
    ch := make(chan int)
    go assign (ch)

    fmt.println(<-ch)
}

“1”被打印出来。

然后我使用了缓冲通道:

package main

import "fmt"

func main() {
    ch := make(chan int, 2)
    ch <- 1
    ch <- 2
    fmt.Println(<-ch)
    fmt.Println(<-ch)
}

“1”和“2”也可以打印出来。

我对目前的情况有点困惑。提前致谢!


解决方案


当您使用无缓冲通道时,goroutine 在写入期间会被阻塞,直到有人执行读取操作。 在您的第一个片段中,有一个无缓冲通道和单个 goroutine(主 goroutine)。 因此,当您尝试编写时:

ch <- 1

尚未有人从频道中读取内容。主 goroutine 被阻塞,这一行永远不会执行:

fmt.println(<-ch)

这就是出现死锁错误的原因。

在第二个示例中,您仍然使用无缓冲通道,这意味着写操作会阻塞 goroutine。 但是通过使用 go 你正在运行第二个 goroutine。 这意味着即使这个新的 goroutine 在写入期间被阻塞(在你的 assign 函数中),主 goroutine 将继续工作,并且 fmt.println(<-ch) 将被执行并执行读取(这反过来又解除阻塞后台 goroutine)而assign函数最终会到达终点)。

为了更多地了解通道和 goroutine,此代码片段将给出相同的结果(与第二个代码片段):

package main

import "fmt"

func print(ch chan int) {
    fmt.println(<-ch)
}

func main() {
    ch := make(chan int)
    go print(ch)
    ch <- 1
}

当您使用缓冲通道(第三个片段)时,您可以执行 n 写入操作,而不会阻塞 goroutine(其中 n 是缓冲区的大小)。 这就是为什么在您的示例中您进行了 2 次写入而没有阻塞,并且能够稍后读取它们。但是,如果您的缓冲区小于写入操作的数量,并且没有人执行读取操作,您将陷入相同的阻塞问题(请参阅1和2片段的解释)。

为什么会发生死锁:

在第一个代码片段中,您只有一个主 goroutine,当您尝试写入此处的通道时,它会被阻止:

ch <- 1

因为没有人从通道读取数据,并且主 goroutine 正在等待继续。

参见Effective Go -> Channels

发送方是main函数,接收方也是main函数。

如何避免死锁:

为了解决这个问题,您有两个选择:

选项 1: 使 ch 通道像这样缓冲:

ch := make(chan int, 1) // buffer length is set to 1

来自A Tour of Go

因此,您可以写入通道,直到缓冲区已满。然后有人必须开始从频道中阅读。

选项 2:从 goroutine 写入通道,就像您在第二个代码片段中所做的那样:

func assign(ch chan int) {
    ch <- 1
}

func main() {
    ch := make(chan int)
    go assign(ch) // does not block the main goroutine
    fmt.Println(<-ch) // waiting to read from the channel
}

在这种情况下,main 函数将被执行,直到 fmt.println(<-ch) 并在可以从通道读取数据后立即继续。

好了,本文到此结束,带大家了解了《比较 Goroutine 中的"go"关键字和"without"》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多Golang知识!

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