目录
前言程序示例select 特性预览管道读写总结前言
在 Go 中,可以通过关键字 select来完成从不同的并发执行的协程中获取值,它和 switch控制语句非常相似,也被称作通信开关;它的行为像是“你准备好了吗”的轮询机制;
(资料图片)
select监听进入通道的数据,也可以是用通道发送值的时候。
select是 Go 在语言层面提供的多路 I/O 复用机制,用于检测多个管道是否就绪(即可读或可写),其特性与管道息息相关。
语法格式:
select {
case u:= <- ch1:
...
case v:= <- ch2:
...
...
default: // no value ready to be received
...
}
default语句是可选的;fallthrough 行为,和普通的 switch 相似,是不允许的。在任何一个 case 中执行 break或者 return,select 就结束了。
select做的就是:选择处理列出的多个通信情况中的一个。
default语句,它就会执行:default永远是可运行的(这就是准备好了,可以执行)。
在 select中使用发送操作并且有 default可以确保发送不被阻塞!如果没有 default,select就会一直阻塞。default不能处理管道读写操作,
select语句实现了一种监听模式,通常用在(无限)循环中;在某种情况下,通过 break语句使循环退出。
程序示例
package main
import (
"fmt"
"time"
)
func main() {
ch1 := make(chan int)
ch2 := make(chan int)
go pump1(ch1)
go pump2(ch2)
go suck(ch1, ch2)
time.Sleep(1e9)
}
func pump1(ch chan int) {
for i := 0; ; i++ {
ch <- i * 2
}
}
func pump2(ch chan int) {
for i := 0; ; i++ {
ch <- i + 5
}
}
func suck(ch1, ch2 chan int) {
for {
select {
case v := <-ch1:
fmt.Printf("Received on channel 1: %d\n", v)
case v := <-ch2:
fmt.Printf("Received on channel 2: %d\n", v)
}
}
}
在程序 goroutine_select.go中有 2 个通道 ch1和 ch2,
三个协程 pump1()、pump2()和 suck()。
这是一个典型的生产者消费者模式。在无限循环中,ch1和 ch2通过 pump1()和 pump2()填充整数;suck()也是在无限循环中轮询输入的,通过 select语句获取 ch1和 ch2的整数并输出。选择哪一个 case 取决于哪一个通道收到了信息。程序在 main 执行 1 秒后结束。
运行结果:
Received on channel 2: 148120
Received on channel 2: 148121
Received on channel 2: 148122
Received on channel 2: 148123
Received on channel 2: 148124
Received on channel 2: 148125
Received on channel 2: 148126
Received on channel 1: 296784
Received on channel 2: 148127
Received on channel 2: 148128
Received on channel 2: 148129
Received on channel 1: 296786
Received on channel 1: 296788
一秒内的输出非常惊人,如果我们给它计数(goroutine_select2.go),得到了 296788 个左右的数字。
select 特性预览
管道读写
select只能作用于管道,包括数据的读取和写入。例如:
package main
import "fmt"
func selectDemo(c chan string) {
recv := ""
send := "Hello"
select {
case recv = <-c:
fmt.Printf("Received %s\n", recv)
case c <- send:
fmt.Printf("Sent %s\n", send)
}
}
如果管道中没有缓存,如下:
func main() {
c := make(chan string)
selectDemo(c)
}
此时管道既不能读也不能写,两个 case 语句都不执行,select陷入阻塞
func main() {
c := make(chan string, 1)
selectDemo(c)
}
此时,管道可以写入,写操作对应的 case 语句得到执行,且执行结束后函数退出。
如果管道有缓冲区且缓冲区中已放满数据,如下:func main() {
c := make(chan string, 1)
c <- "你好,向你说再见!"
selectDemo(c)
}
此时,管道可以读取,读操作对应的 case 语句得到执行,且执行结束后函数退出。
管道有缓冲区,缓冲区中已有部分数据还可以存入数据,如下:func main() {
c := make(chan string, 2)
c <- "你好,向你说再见!"
selectDemo(c)
}
管道的缓冲区有部分且还可以存入数据,此时管道既可以读取也可以写入,select将选取一个 case 语句执行,任意一个 case 语句执行结束后函数就退出。
总结
select 的每个 case 语句只能操作一个管道,要么写入数据,要么读取数据;
如果管道中没有数据读取操作则会阻塞,如果管道中没有空余的缓冲区则写入操作会阻塞;
当 select 的多个 case 语句中的管道均阻塞时,整个 select 语句也会陷入阻塞,直到任意一个管道解除阻塞;
如果多个 case 语句均没有阻塞,那么 select 将随机挑选一个 case 执行。
以上就是Go使用select切换协程入门详解的详细内容,更多关于Go select 切换协程的资料请关注脚本之家其它相关文章!
X 关闭
X 关闭
- 1亚马逊开始大规模推广掌纹支付技术 顾客可使用“挥手付”结账
- 2现代和起亚上半年出口20万辆新能源汽车同比增长30.6%
- 3如何让居民5分钟使用到各种设施?沙特“线性城市”来了
- 4AMD实现连续8个季度的增长 季度营收首次突破60亿美元利润更是翻倍
- 5转转集团发布2022年二季度手机行情报告:二手市场“飘香”
- 6充电宝100Wh等于多少毫安?铁路旅客禁止、限制携带和托运物品目录
- 7好消息!京东与腾讯续签三年战略合作协议 加强技术创新与供应链服务
- 8名创优品拟通过香港IPO全球发售4100万股 全球发售所得款项有什么用处?
- 9亚马逊云科技成立量子网络中心致力解决量子计算领域的挑战
- 10京东绿色建材线上平台上线 新增用户70%来自下沉市场

