当从缓冲通道读取时,所有的goroutine都处于休眠状态。

huangapple go评论144阅读模式
英文:

All goroutines are asleep when reading from buffered channel

问题

  1. func writeToChan(wg *sync.WaitGroup, ch chan int, stop int) {
  2. defer wg.Done()
  3. for i := 0; i < stop; i++ {
  4. ch <- i
  5. }
  6. }
  7. func readToChan(wg *sync.WaitGroup, ch chan int) {
  8. defer wg.Done()
  9. for n := range ch {
  10. fmt.Println(n)
  11. }
  12. }
  13. func main() {
  14. ch := make(chan int, 3)
  15. wg := new(sync.WaitGroup)
  16. wg.Add(2)
  17. go writeToChan(wg, ch, 5)
  18. go readToChan(wg, ch)
  19. wg.Wait()
  20. }
  1. 0
  2. 1
  3. 2
  4. 3
  5. 4
  6. fatal error: all goroutines are asleep - deadlock!

我假设readToChan函数总是连续读取,而writeToChan函数在向通道写入数据后会等待通道被读取。
我不知道为什么输出显示死锁,尽管我已经在WaitGroup中添加了两个wait操作。

英文:
  1. func writeToChan(wg *sync.WaitGroup, ch chan int, stop int) {
  2. defer wg.Done()
  3. for i := 0; i &lt; stop; i++ {
  4. ch &lt;- i
  5. }
  6. }
  7. func readToChan(wg *sync.WaitGroup, ch chan int) {
  8. defer wg.Done()
  9. for n := range ch {
  10. fmt.Println(n)
  11. }
  12. }
  13. func main() {
  14. ch := make(chan int, 3)
  15. wg := new(sync.WaitGroup)
  16. wg.Add(2)
  17. go writeToChan(wg, ch, 5)
  18. go readToChan(wg, ch)
  19. wg.Wait()
  20. }
  1. 0
  2. 1
  3. 2
  4. 3
  5. 4
  6. fatal error: all goroutines are asleep - deadlock!

I assume that the readToChan always reads continuously, and the writeToChan write to the channel and waits while the channel is read.
I don't know why the output showed deadlock while I added two 'wait' to the WaitGroup.

答案1

得分: 2

你需要在发送方关闭通道。
使用以下代码:

  1. for n := range ch {
  2. fmt.Println(n)
  3. }

只有当ch被关闭时,循环才会停止。

正确的示例代码如下:

  1. package main
  2. import (
  3. "fmt"
  4. "sync"
  5. )
  6. func writeToChan(wg *sync.WaitGroup, ch chan int, stop int) {
  7. defer wg.Done()
  8. for i := 0; i < stop; i++ {
  9. ch <- i
  10. }
  11. close(ch)
  12. }
  13. func readToChan(wg *sync.WaitGroup, ch chan int) {
  14. defer wg.Done()
  15. for n := range ch {
  16. fmt.Println(n)
  17. }
  18. }
  19. func main() {
  20. ch := make(chan int, 3)
  21. wg := new(sync.WaitGroup)
  22. wg.Add(2)
  23. go writeToChan(wg, ch, 5)
  24. go readToChan(wg, ch)
  25. wg.Wait()
  26. }
英文:

You need to close channel at the sender side.
By using

  1. for n := range ch {
  2. fmt.Println(n)
  3. }

The loop will only stop when ch is closed

correct example:

  1. package main
  2. import (
  3. &quot;fmt&quot;
  4. &quot;sync&quot;
  5. )
  6. func writeToChan(wg *sync.WaitGroup, ch chan int, stop int) {
  7. defer wg.Done()
  8. for i := 0; i &lt; stop; i++ {
  9. ch &lt;- i
  10. }
  11. close(ch)
  12. }
  13. func readToChan(wg *sync.WaitGroup, ch chan int) {
  14. defer wg.Done()
  15. for n := range ch {
  16. fmt.Println(n)
  17. }
  18. }
  19. func main() {
  20. ch := make(chan int, 3)
  21. wg := new(sync.WaitGroup)
  22. wg.Add(2)
  23. go writeToChan(wg, ch, 5)
  24. go readToChan(wg, ch)
  25. wg.Wait()
  26. }

答案2

得分: 0

如果没有在缓冲通道上调用close函数,读取器就不知道何时停止读取。请查看以下示例,其中使用for和select调用来处理多个通道。

https://go.dev/play/p/Lx5g9o4RsqW

  1. package main
  2. import (
  3. "fmt"
  4. "sync"
  5. "time"
  6. )
  7. func writeToChan(wg *sync.WaitGroup, ch chan int, stop int, quit chan<- bool) {
  8. defer func() {
  9. wg.Done()
  10. close(ch)
  11. fmt.Println("write wg done")
  12. }()
  13. for i := 0; i < stop; i++ {
  14. ch <- i
  15. fmt.Println("write:", i)
  16. }
  17. fmt.Println("write done")
  18. fmt.Println("sleeping for 5 sec")
  19. time.Sleep(5 * time.Second)
  20. quit <- true
  21. close(quit)
  22. }
  23. func readToChan(wg *sync.WaitGroup, ch chan int, quit chan bool) {
  24. defer func() {
  25. wg.Done()
  26. fmt.Println("read wg done")
  27. }()
  28. for {
  29. select {
  30. case n := <-ch:
  31. fmt.Println("read:", n)
  32. case val := <-quit:
  33. fmt.Println("received quit:", val)
  34. return
  35. }
  36. }
  37. }
  38. func main() {
  39. ch := make(chan int, 5)
  40. ch2 := make(chan bool)
  41. wg := new(sync.WaitGroup)
  42. wg.Add(2)
  43. go writeToChan(wg, ch, 3, ch2)
  44. go readToChan(wg, ch, ch2)
  45. wg.Wait()
  46. }

输出:

  1. write: 0
  2. write: 1
  3. write: 2
  4. write done
  5. sleeping for 5 sec
  6. read: 0
  7. read: 1
  8. read: 2
  9. write wg done
  10. received quit: true
  11. read wg done
  12. 程序已退出。
英文:

If close is not called on buffered channel, reader doesn't know when to stop reading.
Check this example with for and select calls(to handle multi channels).

https://go.dev/play/p/Lx5g9o4RsqW

  1. package main
  2. import (
  3. &quot;fmt&quot;
  4. &quot;sync&quot;
  5. &quot;time&quot;)
  6. func writeToChan(wg *sync.WaitGroup, ch chan int, stop int, quit chan&lt;- bool) {
  7. defer func() {
  8. wg.Done()
  9. close(ch)
  10. fmt.Println(&quot;write wg done&quot;)
  11. }()
  12. for i := 0; i &lt; stop; i++ {
  13. ch &lt;- i
  14. fmt.Println(&quot;write:&quot;, i)
  15. }
  16. fmt.Println(&quot;write done&quot;)
  17. fmt.Println(&quot;sleeping for 5 sec&quot;)
  18. time.Sleep(5 * time.Second)
  19. quit &lt;- true
  20. close(quit)
  21. }
  22. func readToChan(wg *sync.WaitGroup, ch chan int, quit chan bool) {
  23. defer func() {
  24. wg.Done()
  25. fmt.Println(&quot;read wg done&quot;)
  26. }()
  27. //using rang over
  28. //for n := range ch {
  29. // fmt.Println(n)
  30. //}
  31. //using Select if you have multiple channels.
  32. for {
  33. //fmt.Println(&quot;waiting for multiple channels&quot;)
  34. select {
  35. case n := &lt;-ch:
  36. fmt.Println(&quot;read:&quot;, n)
  37. // if ok == false {
  38. // fmt.Println(&quot;read done&quot;)
  39. // //return
  40. // }
  41. case val := &lt;-quit:
  42. fmt.Println(&quot;received quit :&quot;, val)
  43. return
  44. // default:
  45. // fmt.Println(&quot;default&quot;)
  46. }
  47. }
  48. }
  49. func main() {
  50. ch := make(chan int, 5)
  51. ch2 := make(chan bool)
  52. wg := new(sync.WaitGroup)
  53. wg.Add(2)
  54. go writeToChan(wg, ch, 3, ch2)
  55. go readToChan(wg, ch, ch2)
  56. wg.Wait()
  57. }

Output:

  1. write: 0
  2. write: 1
  3. write: 2
  4. write done
  5. sleeping for 5 sec
  6. read: 0
  7. read: 1
  8. read: 2
  9. write wg done
  10. received quit : true
  11. read wg done
  12. Program exited.

huangapple
  • 本文由 发表于 2022年8月16日 13:34:52
  • 转载请务必保留本文链接:https://go.coder-hub.com/73369181.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定