Go无缓冲通道的陷阱

原创 2017年08月26日 21:07:54

定义

Channel是go的特色之一,甚至说是最大的特色也不为过,使用起来也非常简单。
首先定义一个int类型的channel:

ch1 := make(chan int) // 无缓冲通道
ch2 := make(chan int, 10) // 带缓冲的通道

我们这里主要关注无缓冲通道。

场景

来看看这段代码:

package main

import (
    "sync"
    "fmt"
)

func main() {
    wg := sync.WaitGroup{}

    ch1 := make(chan int)

    wg.Add(1)

    for i := 0; i < 10; i++ {
        ch1 <- i
    }

    go func1(ch1, &wg) 

    wg.Wait()

    close(ch1)
    fmt.Println("Close channel: ", ch1)
}

func func1(ch chan int, wg *sync.WaitGroup) {
    FOR:
    for {
        select {
        case i, ok := <- ch:
            if !ok {
                fmt.Println("1 chan closed, returning")
                break FOR
            } else {
                fmt.Println("1 Got number: ", i)
                if i == 9 {
                    break FOR
                }
            }
        default:
            fmt.Println("1 Got nothing")
        }
    }
    wg.Done()
}

乍眼一看似乎没毛病,但是当运行程序的时候:

fatal error: all goroutines are asleep - deadlock!

goroutine 1 [chan send]:
main.main()
    /Users/bruce/Code/go/src/GoCommonServices/sync-demo/demo.go:19 +0xb9
exit status 2

Process finished with exit code 1

为什么呢? 仔细看了看上面的程序,在定义了无缓冲通道ch1之后,立马向其中写入数据:

    ch1 := make(chan int)

    for i := 0; i < 10; i++ {
        ch1 <- i
    }

但此时并没有消费者,而无缓冲通道在写入一个数据之后,会等待消费者消费,程序阻塞,但启动消费者的代码:

    go func1(ch1, &wg) 

恰好在for循环之后,所以这个goroutine永远没有启动的机会,这就是报错信息提示的,deadlock了,要修复这个有两种方法:

1 ch1定义为缓冲通道,足够容纳for中的数据,就不会阻塞
    ch1 := make(chan int 10)
2 先启动消费者,再向通道中写数据
    go func1(ch1, &wg) // 先启动消费者,再向ch1中写入数据,以防阻塞

    ch1 := make(chan int)

    for i := 0; i < 10; i++ {
        ch1 <- i
    }

小结

归根结底,还是因为channel的特性:

无缓冲的channel,不管是入还是出,都会阻塞,所以在同一个goroutine中,不能同时对同一个无缓冲channel进行入和出操作;
带缓冲的channel,在队列满之前,不会阻塞;队列满之后,依然会阻塞。

channel无疑是go并发程序开发的利器,但使用的时候还是需要仔细慎重,注意避免像本文提到的这些『陷阱』。

版权声明:转载请署名 欢迎联系博主

go语言通道channel使用总结

1.广播当一个通道关闭时, 所有此通道的读取都会退出阻塞. 利用此特性可以实现广播功能package mainimport ( "fmt" "time" )func main() { ...
  • tsxw24
  • tsxw24
  • 2017年04月03日 21:09
  • 2019

Go语言实战-- 通道

上一篇我们讲的原子函数和互斥锁,都可以保证共享数据的读写,但是呢,它们还是有点复杂,而且影响性能,对此,Go又为我们提供了一种工具,这就是通道。 所以在多个goroutine并发中,我们不仅可以...
  • u013870094
  • u013870094
  • 2017年07月03日 22:20
  • 714

无缓冲通道和死锁

无缓冲通道和死锁
  • h363659487
  • h363659487
  • 2017年12月04日 16:56
  • 82

对golang的Channel初始化的有缓存与无缓存解释

首先编程的时候遇到疑问,输出跟我所想预想不一样,后来查到了golang社区的帖子,其中一篇帖子 :健哥大人  做出了一些解释。 我摘抄重点过来: 无缓冲的与有缓冲channel有着重大差别,那...
  • paladinosment
  • paladinosment
  • 2014年12月29日 21:36
  • 2539

go channel有无缓存的区别

无缓冲的与有缓冲channel有着重大差别,那就是一个是同步的 一个是非同步的。 比如 c1:=make(chan int)         无缓冲 c2:=make(chan int,1)  ...
  • whatday
  • whatday
  • 2017年12月05日 21:25
  • 146

golang的缓冲channel和无缓冲channel的区别

golang channel 有缓冲 与 无缓冲 是有重要区别的 我之前天真的认为 有缓冲与无缓冲的区别 只是 无缓冲的 是 默认 缓冲 为1 的缓冲式 其实是彻底错误的,无缓冲的与有缓冲...
  • samete
  • samete
  • 2016年10月07日 18:58
  • 1269

Go for的三重陷阱

说明这篇for的三重陷阱取自我的另一篇博文Go学习笔记。为了方便快速的查找,将其复制了过来。for陷阱之循环变量 在使用匿名函数时,须特别注意在for词块中引入的循环变量。如果在匿名函数里保存了这个变...
  • u011304970
  • u011304970
  • 2017年05月24日 11:31
  • 763

Go 通道(chan)关闭和后续读取操作

1、通道关闭时间: 2、有/无缓冲通道 3、range遍历通道
  • Tovids
  • Tovids
  • 2017年09月06日 15:41
  • 528

Go语言中的常见陷阱

原文:Common Gotchas in Go 作者:Mike JS. Choi 翻译:雁惊寒 摘要:本文介绍了Go初学者很可能会遇到的三个常见陷阱。以下是译文。 我最近开发了我的第...
  • dev_csdn
  • dev_csdn
  • 2018年01月09日 15:04
  • 477

MySQL缓冲和无缓冲查询对比

MySQL的客户端有两种类型的查询: 缓冲查询:将接收查询的结果并把他们存储在客户端的缓存中,而且接下来获取行记录的请求仅仅从本地内获取。 优点:可以在结果集中自由地移动“当前行”的指针,这样很容...
  • ypb455360299
  • ypb455360299
  • 2012年05月23日 10:16
  • 1138
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Go无缓冲通道的陷阱
举报原因:
原因补充:

(最多只允许输入30个字)