Golang频道睡着了

问题描述 投票:1回答:1

我是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)它工作正常。为什么会这样?

go concurrency deadlock channels
1个回答
3
投票

这是因为主要的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
    }
}
© www.soinside.com 2019 - 2024. All rights reserved.