重庆分公司,新征程启航
为企业提供网站建设、域名注册、服务器等服务
本篇文章为大家展示了Go无缓冲通道的陷阱是什么,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。
10余年的定陶网站建设经验,针对设计、前端、开发、售后、文案、推广等六对一服务,响应快,48小时及时工作处理。网络营销推广的优势是能够根据用户设备显示端的尺寸不同,自动调整定陶建站的显示方式,使网站能够适用不同显示终端,在浏览器中调整网站的宽度,无论在任何一种浏览器上浏览网站,都能展现优雅布局与设计,从而大程度地提升浏览体验。成都创新互联公司从事“定陶网站设计”,“定陶网站推广”以来,每个客户项目都认真落实执行。
Channel是go的特色之一,甚至说是最大的特色也不为过,使用起来也非常简单。
首先定义一个int类型的channel:
ch2 := make(chan int) ch3 := make(chan int, 10)
我们这里主要关注无缓冲通道。
来看看这段代码:
package mainimport ( "sync" "fmt")func main() { wg := sync.WaitGroup{} ch2 := make(chan int) wg.Add(1) for i := 0; i < 10; i++ { ch2 <- i } go func1(ch2, &wg) wg.Wait() close(ch2) fmt.Println("Close channel: ", ch2) }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 +0xb9exit status 2Process finished with exit code 1
为什么呢? 仔细看了看上面的程序,在定义了无缓冲通道ch2之后,立马向其中写入数据:
ch2 := make(chan int) for i := 0; i < 10; i++ { ch2 <- i }
但此时并没有消费者,而无缓冲通道在写入一个数据之后,会等待消费者消费,程序阻塞,但启动消费者的代码:
go func1(ch2, &wg)
恰好在for循环之后,所以这个goroutine永远没有启动的机会,这就是报错信息提示的,deadlock了,要修复这个有两种方法:
ch2 := make(chan int, 10)
go func1(ch2, &wg) ch2 := make(chan int) for i := 0; i < 10; i++ { ch2 <- i }
归根结底,还是因为channel的特性:
无缓冲的channel,不管是入还是出,都会阻塞,所以在同一个goroutine中,不能同时对同一个无缓冲channel进行入和出操作;
带缓冲的channel,在队列满之前,不会阻塞;队列满之后,依然会阻塞。
golang是一种编译语言,可以将代码编译为机器代码,编译后的二进制文件可以直接部署到目标机器而无需额外的依赖,所以golang的性能优于其他的解释性语言,且可以在golang中使用goroutine来实现并发性,它提供了一个非常优雅的goroutine调度程序系统,可以很容易地生成数百万个goroutine。
上述内容就是Go无缓冲通道的陷阱是什么,你们学到知识或技能了吗?如果还想学到更多技能或者丰富自己的知识储备,欢迎关注创新互联行业资讯频道。