select选择器+go协程+channel通道的组合
上面讲过阻塞的原理,以及一些可以用到的场景
如果我有多个协程在运行,有的快就的慢.
那么可以用selsect 同时等.
select选择器+go协程+channel通道的组合 是一个强大的特性
package main
import "time"
import "fmt"
func main() {
c1 := make(chan string)
c2 := make(chan string)
go func() {
time.Sleep(time.Second * 1)
c1 <- "one"
}()
go func() {
time.Sleep(time.Second * 2)
c2 <- "two"
}()
for i := 0; i < 2; i++ {
select {
case msg1 := <-c1:
fmt.Println("received", msg1)
case msg2 := <-c2:
fmt.Println("received", msg2)
}
}
}
select还可以用于超时处理
这是在服务器稳定性上 是比较实用的
上面例子,如果协程很久都没有数据过来,或者协程阻塞在某处,我们得设计好超时时间,否则我们也将无法工作.那么可以看下面的处理
package main
import "time"
import "fmt"
func main() {
c1 := make(chan string, 1)
go func() {
time.Sleep(time.Second * 2)
c1 <- "result 1"
}()
select {
case res := <-c1:
fmt.Println(res)
case <-time.After(time.Second * 1):
fmt.Println("timeout 1")
}
}
/*
我们设定了1秒的超时时间,而协程2秒后才传送过来,那么该数据我们直接抛弃了
*/
//用default 也可 没有数据的情况下直接跳过
select {
case res := <-c1:
fmt.Println("获得数据", msg)
default:
fmt.Println("直接跳过")
}