登录
首页 >  Golang >  Go问答

“对于 goroutine 泄漏,是否会受到“done”通道和默认情况的影响?”

来源:stackoverflow

时间:2024-02-16 10:00:26 436浏览 收藏

小伙伴们有没有觉得学习Golang很有意思?有意思就对了!今天就给大家带来《“对于 goroutine 泄漏,是否会受到“done”通道和默认情况的影响?”》,以下内容将会涉及到,若是在学习中对其中部分知识点有疑问,或许看了本文就能帮到你!

问题内容

我想与您比较两个类似的案例 - 唯一的区别是处理值生成的方式

  • 第一种情况:在 select 的一个情况下生成值
package main

import (
    "fmt"
    "math/rand"
    "time"
)

func main() {
    generatevalues := func(done <-chan interface{}) <-chan int {
        values := make(chan int)
        go func() {
            defer fmt.println("all values generated")
            defer close(values)
            for {
                select {
                case <-done:
                    fmt.println("done")
                    return
                case values <- rand.int():
                    fmt.println("generated")
                }

            }

        }()
        return values
    }

    done := make(chan interface{})
    values := generatevalues(done)

    for i := 0; i < 3; i++ {
        fmt.printf("received value: %v\n", <-values)
    }
    fmt.println("closing the channel")
    close(done)
    time.sleep(5 * time.second)
}

go 演示:https://go.dev/play/p/edlosqdz9ys

  • 第二种情况:默认情况下的值生成
package main

import (
    "fmt"
    "math/rand"
    "time"
)

func main() {
    generateValues := func(done <-chan interface{}) <-chan int {
        values := make(chan int)
        go func() {
            defer fmt.Println("All values generated")
            defer close(values)
            for {
                select {
                case <-done:
                    fmt.Println("DONE")
                    return
                default:
                    values <- rand.Int()
                    fmt.Println("Generated")
                }

            }

        }()
        return values
    }

    done := make(chan interface{})
    values := generateValues(done)

    for i := 0; i < 3; i++ {
        fmt.Printf("Received value: %v\n", <-values)
    }
    fmt.Println("Closing the channel")
    close(done)
    time.Sleep(5 * time.Second)
}

go 演示:https://go.dev/play/p/edlosqdz9ys

正如您所看到的,第二种情况似乎会导致未打印“完成”并且未调用与“延迟”相关的调用的情况。我相信我们这里有 goroutine 泄漏,但无法清楚地解释它。我预计会出现与第一种情况相同的行为。

有人可以帮助理解它们之间的区别吗?


正确答案


在第二种情况下,生成协程不太可能收到 done 消息。由于 default 情况始终处于启用状态,因此主 Goroutine 接收到最后一个值后,生成 Goroutine 会进行另一轮,陷入 default 情况,并阻塞等待写入 values 通道。当它在那里等待时,主协程关闭 done 通道并终止。

这并不意味着不存在生成 goroutine 不接收已完成通道的执行路径。为此,在发送最后一个值后,生成 Goroutine 必须立即被运行的主 Goroutine 抢占,直到它关闭 done 通道。那么如果生成协程被调度,它就可以接收到 done 信号。然而,这一系列事件的可能性极小。

本篇关于《“对于 goroutine 泄漏,是否会受到“done”通道和默认情况的影响?”》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注golang学习网公众号!

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