登录
首页 >  Golang >  Go问答

延迟函数为何未被触发?

来源:stackoverflow

时间:2024-02-26 21:45:32 396浏览 收藏

偷偷努力,悄无声息地变强,然后惊艳所有人!哈哈,小伙伴们又来学习啦~今天我将给大家介绍《延迟函数为何未被触发?》,这篇文章主要会讲到等等知识点,不知道大家对其都有多少了解,下面我们就一起来看一吧!当然,非常希望大家能多多评论,给出合理的建议,我们一起学习,一起进步!

问题内容

我正在努力理解 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

根据我的理解,defer函数将在其周围函数完成后执行。我无法解释它。

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

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

谁能帮我理解这个概念?


解决方案


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

用稍微不同的术语来解释它,您的代码的作用是:

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
    }
}

本篇关于《延迟函数为何未被触发?》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注golang学习网公众号!

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