select {
case <-chan1:
// 如果chan1成功读到数据,则进行该case处理语句
case chan2 <- 1:
// 如果成功向chan2写入数据,则进行该case处理语句
default:
// 如果上面都没有成功,则进入default处理流程
}*/
只要是想要在多个通道上读取或者写入数据,就用select,但是注意,select和switch-case是一样的,整个select代码段只执行一次,比如下面的代码段,jobChan和resultChan中一旦无法读取到内容的话,直接进入default,然后整个select代码段就执行结束了
jobChan := make(chan int , 10)
resultChan := make(chan float32, 20)
//和for不一样,for是永远循环,但是select和switch-case很类似,只要探测到某个分支满足条件,就会执行对应的case,然后就退出了
select {
case id, ok := <-jobChan:
fmt.Println(id, ok)
case result, ok := <- resultChan:
fmt.Println(result, ok)
default:
fmt.Println("enter default statement")
}
但是,如果不加default分支,select会一直等,比如下面的代码,直到output1和output2这两个通道任意一个能够读取到数据,就执行对应的case语句,然后就结束整个select代码段的执行
func test1(ch chan string) {
time.Sleep(time.Second * 5)
ch <- "test1"
}
func test2(ch chan string) {
time.Sleep(time.Second * 2)
ch <- "test2"
}
func TestSelectKeyword(t *testing.T) {
// 2个管道
output1 := make(chan string)
output2 := make(chan string)
// 跑2个子协程,写数据
go test1(output1)
go test2(output2)
// 用select监控
select {
case s1 := <-output1:
fmt.Println("s1=", s1)
case s2 := <-output2:
fmt.Println("s2=", s2)
}
}
这个纯粹是语法规则,记住即可,尤其是select是用于检测通道中是否可以读取数据的,它用于多通道检测。