Golang频道睡着了

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Golang频道睡着了相关的知识,希望对你有一定的参考价值。

我是golang频道的新手,如果指定通道缓冲区大小,我不能低估为什么程序行为会发生变化

package main

import (
    "fmt"
)

func channels(in <-chan bool, out chan int) {
    for {
        select {
        case _ = <-in:
            fmt.Println("Close")
            close(out)
            return
        default:
            fmt.Println("Out")
            out <- 1
            break
        }
    }
}

func main() {
    in := make(chan bool)
    // in := make(chan bool, 1)
    out := make(chan int)

    go channels(in, out)

    i := 0

    // Loop:
    for n := range out {
        fmt.Println(n)

        i += 1

        if i > 10 {
            fmt.Println("Send close")
            in <- true // Here it is became asleep
            return
        }
    }

    fmt.Println("Done")
}

输出是:

Out
Out
1
1
Out
Out
1
1
Out
Out
1
1
Out
Out
1
1
Out
Out
1
1
Out
Out
1
Send close
fatal error: all goroutines are asleep - deadlock!

goroutine 1 [chan send]:
main.main()
    /home/user/go-lessons/3/chan_4.go:39 +0x2c3

goroutine 19 [chan send]:
main.channels(0xc820072060, 0xc8200720c0)
    /home/user/go-lessons/3/chan_4.go:16 +0x241
created by main.main
    /home/user/go-lessons/3/chan_4.go:27 +0x97
exit status 2

如果我用in := make(chan bool)替换in := make(chan bool, 1)它工作正常。为什么会这样?

答案

这是因为主要的goroutine被写入in

in <- true

另一个goroutine被卡在写给out

out <- 1

如果你把写也放case它应该工作:

for {
    select {
    case <-in:
        fmt.Println("Close")
        close(out)
        return
    case out <- 1:
        fmt.Println("Out")
        break
    }
}

以上是关于Golang频道睡着了的主要内容,如果未能解决你的问题,请参考以下文章

golang代码片段(摘抄)

了解golang频道:死锁

golang goroutine例子[golang并发代码片段]

代码片段 - Golang 实现简单的 Web 服务器

在 go 中输入不可知的频道

代码片段 - Golang 实现集合操作