通道技术和中断技术(Go 快速入门指南 – 缓冲通道和非缓冲通道)
概述
建议先阅读goroutine小节 。
Go 箴言: 不要通过共享内存来通信 ,而要通过通信来共享内存 。
goroutine是 Go 程序并发执行的实体 ,channel (通道)则是它们之间的连接 ,用于多个goroutine之间互相通信 。通道可以让一个goroutine发送特定类型值到另一个goroutine ,每一个通道可以发送数据类型称为通道的元素类型 。
阻塞通道与非阻塞通道
通过关键字chan+数据类型来表明通道数据类型 ,调用make()函数来初始化一个通道 。make()函数的第二个参数为通道长度 ,如果未指定或指定为 0 ,则该通道为非缓存通道 (阻塞通道), 否则该通道为缓存通道 (非阻塞通道) 。
阻塞通道
图片来源: https://stackoverflow.com/questions/39826692/what-are-channels-used-for
例子
ch := make(chan string) // 非缓冲通道 ch := make(chan string, 0) // 非缓冲通道 ch := make(chan string, 10) // 缓冲通道, 容量为 103 种操作
发送
无缓冲通道上面的发送操作将会阻塞 ,直到另一个goroutine在对应的通道上面完成接收操作 ,两个goroutine才可以继续执行 。
语法规则
通道变量<-数据 #例如:将变量x发送到通道ch ch<-x接收
无缓冲通道上面的接收操作将会阻塞,直到另一个goroutine在对应的通道上面完成发送操作 ,两个goroutine才可以继续执行 。
语法规则
<-通道变量 #例如:从通道ch接收一个值 ,并且丢弃 <-ch 接收变量<-通道变量 #例如:从通道ch接收一个值,并且赋值给变量x x:=<-ch关闭
详情见关闭通道 。
例子
搭配 goroutine
packagemain funcmain(){ ch:=make(chanstring)//没有设置通道的长度 gofunc(){ ch<-"helloworld" }() msg:=<-ch//一直阻塞 ,直到接收到通道消息 println(msg) } //$gorunmain.go //输出如下 /** helloworld */死锁
packagemain funcmain(){ ch:=make(chanstring)//没有设置通道的长度 ch<-"helloworld"//向通道发送数据 ,但是没有接收者 msg:=<-ch//代码执行不到这里,因为上面阻塞发送数据时,就已经死锁了 println(msg) } //$gorunmain.go //输出如下 /** fatalerror:allgoroutinesareasleep-deadlock! ... ... exitstatus2 */非阻塞通道
图片来源: https://stackoverflow.com/questions/39826692/what-are-channels-used-for
例子
ch:=make(chanstring,10)//缓冲通道,容量为103 种操作
发送
•如果通道已满 (元素数量达到容量), 发送操作将会阻塞 ,直到另一个goroutine在对应的通道上面完成接收操作 , 两个goroutine才可以继续执行
•如果通道未满 ,发送操作不会阻塞
语法规则
通道变量<-数据 #例如:将变量x发送到通道ch ch<-x接收
•如果通道已空 (元素数量为 0) ,接收操作将会阻塞 ,直到另一个goroutine在对应的通道上面完成发送操作 , 两个goroutine才可以继续执行
•如果通道不为空 ,接收操作不会阻塞
语法规则
<-通道变量 #例如:从通道ch接收一个值 ,并且丢弃 <-ch 接收变量<-通道变量 #例如:从通道ch接收一个值 ,并且赋值给变量x x:=<-ch关闭
详情见关闭通道。
例子
缓存通道容量为 2
packagemain funcmain(){ ch:=make(chanstring,2) ch<-"hello"//不会死锁,因为ch是缓冲通道 ch<-"world" println(<-ch) println(<-ch) } //$gorunmain.go //输出如下 /** hello world */扩展阅读
====
1.死锁 - 维基百科联系我
创心域SEO版权声明:以上内容作者已申请原创保护,未经允许不得转载,侵权必究!授权事宜、对本内容有异议或投诉,敬请联系网站管理员,我们将尽快回复您,谢谢合作!