在线时间:8:00-16:00
迪恩网络APP
随时随地掌握行业动态
扫描二维码
关注迪恩网络微信公众号
golang中的select语句格式如下 select { case <-ch1: // 如果从 ch1 信道成功接收数据,则执行该分支代码 case ch2 <- 1: // 如果成功向 ch2 信道成功发送数据,则执行该分支代码 default: // 如果上面都没有成功,则进入 default 分支处理流程 }
可以看到select的语法结构有点类似于switch,但又有些不同。 select里的case后面并不带判断条件,而是一个信道的操作,不同于switch里的case,对于从其它语言转过来的开发者来说有些需要特别注意的地方。 golang 的 select 就是监听 IO 操作,当 IO 操作发生时,触发相应的动作每个case语句里必须是一个IO操作,确切的说,应该是一个面向channel的IO操作。
注意:如果 知识点
下面列出每种情况的示例代码 1. select语句只能用于信道的读写操作package main import "fmt" func main() { size := 10 ch := make(chan int, size) for i := 0; i < size; i++ { ch <- 1 } ch2 := make(chan int, size) for i := 0; i < size; i++ { ch2 <- 2 } ch3 := make(chan int, 1) select { case 3 == 3: fmt.Println("equal") case v := <-ch: fmt.Print(v) case b := <-ch2: fmt.Print(b) case ch3 <- 10: fmt.Print("write") default: fmt.Println("none") } } 语句会报错 prog.go:20:9: 3 == 3 evaluated but not used prog.go:20:9: select case must be receive, send or assign recv
2. select中的case语句是随机执行的package main import "fmt" func main() { size := 10 ch := make(chan int, size) for i := 0; i < size; i++ { ch <- 1 } ch2 := make(chan int, size) for i := 0; i < size; i++ { ch2 <- 2 } ch3 := make(chan int, 1) select { case v := <-ch: fmt.Print(v) case b := <-ch2: fmt.Print(b) case ch3 <- 10: fmt.Print("write") default: fmt.Println("none") } } 多次执行的话,会随机输出不同的值,分别为1,2,write。这是因为ch和ch2是并发执行会同时返回数据,所以会随机选择一个case执行,。但永远不会执行default语句,因为上面的三个case都是可以操作的信道。
3. 对于case条件语句中,如果存在通道值为nil的读写操作,则该分支将被忽略package main import "fmt" func main() { var ch chan int // ch = make(chan int) go func(c chan int) { c <- 100 }(ch) select { case <-ch: fmt.Print("ok") } } 报错 fatal error: all goroutines are asleep - deadlock! goroutine 1 [select (no cases)]: main.main() /tmp/sandbox488456896/main.go:14 +0x60 goroutine 5 [chan send (nil chan)]: main.main.func1(0x0, 0x1043a070) /tmp/sandbox488456896/main.go:10 +0x40 created by main.main /tmp/sandbox488456896/main.go:9 +0x40 可以看到 “goroutine 1 [select (no cases)]” ,虽然写了case条件,但操作的是nil通道,被优化掉了。 要解决这个问题,只能使用make()进行初始化才可以。
4. 超时用法package main import ( "fmt" "time" ) func main() { ch := make(chan int) go func(c chan int) { // 修改时间后,再查看执行结果 time.Sleep(time.Second * 1) ch <- 1 }(ch) select { case v := <-ch: fmt.Print(v) case <-time.After(2 * time.Second): // 等待 2s fmt.Println("no case ok") } time.Sleep(time.Second * 10) } 我们通过修改上面的时等待时间可以看到,如果等待时间超出<2秒,则输出1,否则打印“no case ok”
5. 空select{}package main func main() { select {} } goroutine 1 [select (no cases)]: main.main() /root/project/practice/mytest/main.go:10 +0x20 exit status 2 直接死锁
6. for中的select 引起的CPU过高的问题package main import ( "runtime" "time" ) func main() { quit := make(chan bool) for i := 0; i != runtime.NumCPU(); i++ { go func() { for { select { case <-quit: break default: } } }() } time.Sleep(time.Second * 15) for i := 0; i != runtime.NumCPU(); i++ { quit <- true } }
上面这段代码会把所有CPU都跑满,原因就就在 一般来说,我们用 解决方法就是把
|
请发表评论