协程死锁?

标签 go deadlock goroutine

我是 golang 的新手,我对这个死锁感到困惑 (run here)

package main

import (
    "fmt"
    "runtime"
    "time"
)

func main() {
    c := make(chan string)
    work := make(chan int, 1)
    clvl := runtime.NumCPU()
    count := 0
    for i := 0; i < clvl; i++ {
        go func(i int) {
            for jdId := range work {
                time.Sleep(time.Second * 1)
                c <- fmt.Sprintf("done %d", jdId)
            }
        }(i)
    }

    go func() {
        for i := 0; i < 10; i++ {
            work <- i
        }

        close(work)
    }()

    for resp := range c {
        fmt.Println(resp, count)
        count += 1
    }
}

最佳答案

您永远不会关闭 c,因此您的 for range 循环将永远等待。像这样关闭它:

var wg sync.WaitGroup
for i := 0; i < clvl; i++ {
    wg.Add(1)
    go func(i int) {
        defer wg.Done()
        for jdId := range work {
            time.Sleep(time.Second * 1)
            c <- fmt.Sprintf("done %d", jdId)
        }
    }(i)
}

go func() {
    for i := 0; i < 10; i++ {
        work <- i
    }

    close(work)
    wg.Wait()
    close(c)
}()

编辑:修复了 panic ,感谢 Crast

关于协程死锁?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34807804/

相关文章:

go - 从 dockerize golang 模板中的文件内容设置变量

python - 模拟线程上的死锁

go - 是否关闭 channel 阻塞直到接收者读取它

Golang 非阻塞 channel 不起作用

go - 使用 "/"字符修补 kubernetes 标签

go - 无人机日志存储在哪里?

java - 为什么在静态初始化器中使用并行流会导致不稳定的死锁

java - 如何避免容易陷入僵局?

go - 同时等待套接字连接和 channel

go - 如何在 go 中获取电子邮件正文(使用 imap )?