什么是select?
select是Golang中的一个关键字,用来监听channel相关的IO操作,当IO操作发生时,触发相应的动作。select有如下特点:
- select只能用于channel的操作,每个case分支都只能包含channel的操作表达式;
- select既可以用于channel的数据接收,也可用于channel的数据发送;
- select默认阻塞,只有监听到channel中有发送或者接收数据时才运行;
- 如果设置了default分支则不阻塞,没有监听到channel收发数据则执行default分支;
- 如果不加default分支,有产生死锁的风险;
- select的多个分支都满足条件时,会随机选取其中一个分支执行;
- selec没有任何分支或有分支但没有任何分支满足条件的情况下会永远阻塞。
// 没有任何分支
select{}
//虽然有多分支,但没有分支满足条件
ch1 := make(chan int)
ch2 := make(chan int)
select {
case <-ch1:
fmt.Println("received from ch1")
case <-ch2:
fmt.Println("received from ch2")
}
select基本使用方法
select的基本语法如下:
select {
case x := <- ch1:
// 处理从ch1中接收到的数据x
case y := <- ch2:
// 处理从ch2中接收到的数据y
default:
// 如果所有的channel都没有数据可读,则执行默认操作
}
本例中,使用select监听了多个channel,并使用不同的case语句分别处理不同的channel。如果多个case都可以执行,则会随机选择其中一个执行。如果所有的case都不能执行,则会执行default。
select语句用于监听向channel发送数据的例子如下:
select {
case ch1 <- x:
// 向ch1中发送数据x
case ch2 <- y:
// 向ch2中发送数据y
default:
// 如果所有的channel都已满,则执行默认操作
}
再看个多个分支都满足的例子:
package main
import "fmt"
func main() {
ch1 := make(chan int, 3)
ch1 <- 1
ch2 := make(chan int, 3)
ch2 <- 2
select {
case <-ch1:
fmt.Println("received from ch1")
case <-ch2:
fmt.Println("received from ch2")
}
}
本例中,两个case分支都满足条件,会随机选择一个分支执行,可以自己多运行几次代码观察效果。