话说golang的channel同步的定义真是让人无力吐槽,码农的用户体验就这么难搞么,超耐磨阿,无缓冲和缓冲居然有这么大区别。。。。靠
转载一段网上的资料
--------------------------------------------------------------------------------------------------------------------------------------------------------
golang channel 有缓冲 与 无缓冲 是有重要区别的
我之前天真的认为 有缓冲与无缓冲的区别 只是 无缓冲的 是 默认 缓冲 为1 的缓冲式
其实是彻底错误的,无缓冲的与有缓冲channel有着重大差别
那就是一个是同步的 一个是非同步的
怎么说?比如
c1:=make(chan int) 无缓冲
c2:=make(chan int,1) 有缓冲
c1<-1
无缓冲的 不仅仅是 向 c1 通道放 1 而是 一直要有别的携程 <-c1 接手了 这个参数,那么c1<-1才会继续下去,要不然就一直阻塞着
而 c2<-1 则不会阻塞,因为缓冲大小是1 只有当 放第二个值的时候 第一个还没被人拿走,这时候才会阻塞。
打个比喻
无缓冲的 就是一个送信人去你家门口送信 ,你不在家 他不走,你一定要接下信,他才会走。
无缓冲保证信能到你手上
有缓冲的 就是一个送信人去你家仍到你家的信箱 转身就走 ,除非你的信箱满了 他必须等信箱空下来。
有缓冲的 保证 信能进你家的邮箱
看测试代码
有缓冲的
1 package main 2 3 import "fmt" 4 5 var c = make(chan int, 1) 6 7 func f() { 8 9 c <- 'c' 10 11 fmt.Println("在goroutine内") 12 } 13 14 func main() { 15 go f() 16 17 c <- 'c' 18 <-c 19 <-c 20 21 fmt.Println("外部调用") 22 }
无缓冲的
// pp package main import ( "fmt" ) func writeRoutine(test_chan chan int, value int) { test_chan <- value } func readRoutine(test_chan chan int) { <-test_chan return } func main() { c := make(chan int) x := 100 //readRoutine(c) //go writeRoutine(c, x) //writeRoutine(c, x) //go readRoutine(c) //go readRoutine(c) //writeRoutine(c, x) go writeRoutine(c, x) readRoutine(c) fmt.Println(x) }
可以在部分代码注释掉,看效果,再体会下