[英]fatal error: all goroutines are asleep - deadlock! , This error i am getting
[英]Why am I getting all goroutines are asleep when I close the channel after waiting?
以下是代码:
func makeData() map[string][]Data {
m := make(map[string][]Data)
s := "abcdefghijklmno"
for i, c := range s {
data := []Data{
{value: "hey_" + string(c), id: i * i},
{value: "hello_" + string(c) + string(c), id: i + i},
{value: "bye_" + string(c), id: i + 1},
}
m[strconv.Itoa(i)] = data
}
return m
}
func process(key string, value []Data) (*Result, error) {
if key == "hey_a" {
return nil, errors.New("error")
}
res := Result{data: Data{value: "hi", id: 0}, id: 1}
return &res, nil
}
func main() {
runtime.GOMAXPROCS(runtime.NumCPU())
m := makeData()
errg := new(errgroup.Group)
mapChan := make(chan StringAndData)
sliceChan := make(chan *Result)
for key, value := range m {
key := key
value := value
errg.Go(func() error {
return func(key string, value []Data) error {
res, err := process(key, value)
if err != nil {
return err
}
if res == nil {
return nil
}
if res.data.id == 1 {
mapChan <- StringAndData{
str: key,
data: res.data,
}
return nil
}
sliceChan <- res
return nil
}(key, value)
})
}
if err := errg.Wait(); err != nil {
fmt.Println("error")
} else {
fmt.Println("success")
}
close(mapChan)
close(sliceChan)
for ac := range mapChan {
fmt.Println(ac.str)
}
}
type Data struct {
value string
id int
}
type Result struct {
data Data
id int
}
type StringAndData struct {
str string
data Data
}
我收到fatal error: all goroutines are asleep - deadlock!
但是我在errg.Wait()
之后关闭了频道并且无法理解原因。
我正在尝试打印使用range
关闭通道后从通道获得的值。
我是 go 的通道和并发新手,我将不胜感激!
编辑从操场链接添加了所有代码
查看您的代码,有两件事可能导致死锁:
errg.Wait()
阻塞主 goroutine 的执行,直到所有初始化的 goroutine 完成。 但是,每个 goroutine 在尝试写入mapChan
时都会被阻止,因为您永远无法读取它(因为它位于errg.Wait()
下方)。sliceChan
,所以这是一个潜在的死锁。 这是修改后的 Playground 代码的链接,但大部分更改都在main
的 function 中。
func main() {
runtime.GOMAXPROCS(runtime.NumCPU())
m := makeData()
errg := new(errgroup.Group)
mapChan := make(chan StringAndData)
sliceChan := make(chan *Result)
mapDone := make(chan bool)
sliceDone := make(chan bool)
go func(){
for ac := range mapChan {
fmt.Println(ac.str)
}
mapDone <- true
}()
go func(){
for ac := range sliceChan {
fmt.Println(ac)
}
sliceDone <- true
}()
for key, value := range m {
key := key
value := value
errg.Go(func() error {
return func(key string, value []Data) error {
res, err := process(key, value)
if err != nil {
return err
}
if res == nil {
return nil
}
if res.data.id == 1 {
mapChan <- StringAndData{
str: key,
data: res.data,
}
return nil
}
sliceChan <- res
return nil
}(key, value)
})
}
if err := errg.Wait(); err != nil {
fmt.Println("error")
} else {
fmt.Println("success")
}
close(mapChan)
close(sliceChan)
<-mapDone
<-sliceDone
fmt.Println("finished")
}
基本上,我改变了从mapChan
和sliceChan
通道读取值的方式。 这是在单独的 goroutine 中完成的,因此从这些通道中读取不会被阻塞。
添加mapDone
和sliceDone
通道只是为了确保在main
goroutine 完成之前读取所有数据。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.