Golang中的Channel詳解:Channel與Select
什么是select?
select是Golang中的一個(gè)關(guān)鍵字,用來監(jiān)聽channel相關(guān)的IO操作,當(dāng)IO操作發(fā)生時(shí),觸發(fā)相應(yīng)的動(dòng)作。select有如下特點(diǎn):
- select只能用于channel的操作,每個(gè)case分支都只能包含channel的操作表達(dá)式;
- select既可以用于channel的數(shù)據(jù)接收,也可用于channel的數(shù)據(jù)發(fā)送;
- select默認(rèn)阻塞,只有監(jiān)聽到channel中有發(fā)送或者接收數(shù)據(jù)時(shí)才運(yùn)行;
- 如果設(shè)置了default分支則不阻塞,沒有監(jiān)聽到channel收發(fā)數(shù)據(jù)則執(zhí)行default分支;
- 如果不加default分支,有產(chǎn)生死鎖的風(fēng)險(xiǎn);
- select的多個(gè)分支都滿足條件時(shí),會(huì)隨機(jī)選取其中一個(gè)分支執(zhí)行;
- selec沒有任何分支或有分支但沒有任何分支滿足條件的情況下會(huì)永遠(yuǎn)阻塞。
// 沒有任何分支
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中接收到的數(shù)據(jù)x
case y := <- ch2:
// 處理從ch2中接收到的數(shù)據(jù)y
default:
// 如果所有的channel都沒有數(shù)據(jù)可讀,則執(zhí)行默認(rèn)操作
}
本例中,使用select監(jiān)聽了多個(gè)channel,并使用不同的case語句分別處理不同的channel。如果多個(gè)case都可以執(zhí)行,則會(huì)隨機(jī)選擇其中一個(gè)執(zhí)行。如果所有的case都不能執(zhí)行,則會(huì)執(zhí)行default。
select語句用于監(jiān)聽向channel發(fā)送數(shù)據(jù)的例子如下:
select {
case ch1 <- x:
// 向ch1中發(fā)送數(shù)據(jù)x
case ch2 <- y:
// 向ch2中發(fā)送數(shù)據(jù)y
default:
// 如果所有的channel都已滿,則執(zhí)行默認(rèn)操作
}
再看個(gè)多個(gè)分支都滿足的例子:
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")
}
}
本例中,兩個(gè)case分支都滿足條件,會(huì)隨機(jī)選擇一個(gè)分支執(zhí)行,可以自己多運(yùn)行幾次代碼觀察效果。