Go使用select切換協(xié)程入門詳解
前言
在 Go 中,可以通過關(guān)鍵字 select
來完成從不同的并發(fā)執(zhí)行的協(xié)程中獲取值,它和 switch
控制語句非常相似,也被稱作通信開關(guān);它的行為像是“你準(zhǔn)備好了嗎”的輪詢機(jī)制;
select
監(jiān)聽進(jìn)入通道的數(shù)據(jù),也可以是用通道發(fā)送值的時候。
select
是 Go 在語言層面提供的多路 I/O 復(fù)用機(jī)制,用于檢測多個管道是否就緒(即可讀或可寫),其特性與管道息息相關(guān)。
語法格式:
select { case u:= <- ch1: ... case v:= <- ch2: ... ... default: // no value ready to be received ... }
default
語句是可選的;fallthrough 行為,和普通的 switch 相似,是不允許的。在任何一個 case 中執(zhí)行 break
或者 return
,select 就結(jié)束了。
select
做的就是:選擇處理列出的多個通信情況中的一個。
- 如果都阻塞了,會等待直到其中一個可以處理
- 如果多個可以處理,隨機(jī)選擇一個
- 如果沒有通道操作可以處理并且寫了
default
語句,它就會執(zhí)行:default
永遠(yuǎn)是可運(yùn)行的(這就是準(zhǔn)備好了,可以執(zhí)行)。
在 select
中使用發(fā)送操作并且有 default
可以確保發(fā)送不被阻塞!如果沒有 default
,select
就會一直阻塞。default
不能處理管道讀寫操作,
select
語句實(shí)現(xiàn)了一種監(jiān)聽模式,通常用在(無限)循環(huán)中;在某種情況下,通過 break
語句使循環(huán)退出。
程序示例
package main import ( "fmt" "time" ) func main() { ch1 := make(chan int) ch2 := make(chan int) go pump1(ch1) go pump2(ch2) go suck(ch1, ch2) time.Sleep(1e9) } func pump1(ch chan int) { for i := 0; ; i++ { ch <- i * 2 } } func pump2(ch chan int) { for i := 0; ; i++ { ch <- i + 5 } } func suck(ch1, ch2 chan int) { for { select { case v := <-ch1: fmt.Printf("Received on channel 1: %d\n", v) case v := <-ch2: fmt.Printf("Received on channel 2: %d\n", v) } } }
在程序 goroutine_select.go
中有 2 個通道 ch1
和 ch2
,
三個協(xié)程 pump1()
、pump2()
和 suck()
。
這是一個典型的生產(chǎn)者消費(fèi)者模式。在無限循環(huán)中,ch1
和 ch2
通過 pump1()
和 pump2()
填充整數(shù);suck()
也是在無限循環(huán)中輪詢輸入的,通過 select
語句獲取 ch1
和 ch2
的整數(shù)并輸出。選擇哪一個 case 取決于哪一個通道收到了信息。程序在 main 執(zhí)行 1 秒后結(jié)束。
運(yùn)行結(jié)果:
Received on channel 2: 148120
Received on channel 2: 148121
Received on channel 2: 148122
Received on channel 2: 148123
Received on channel 2: 148124
Received on channel 2: 148125
Received on channel 2: 148126
Received on channel 1: 296784
Received on channel 2: 148127
Received on channel 2: 148128
Received on channel 2: 148129
Received on channel 1: 296786
Received on channel 1: 296788
一秒內(nèi)的輸出非常驚人,如果我們給它計(jì)數(shù)(goroutine_select2.go),得到了 296788 個左右的數(shù)字。
select 特性預(yù)覽
管道讀寫
select
只能作用于管道,包括數(shù)據(jù)的讀取和寫入。例如:
package main import "fmt" func selectDemo(c chan string) { recv := "" send := "Hello" select { case recv = <-c: fmt.Printf("Received %s\n", recv) case c <- send: fmt.Printf("Sent %s\n", send) } }
- 如果管道中沒有緩存,如下:
func main() { c := make(chan string) selectDemo(c) }
此時管道既不能讀也不能寫,兩個 case 語句都不執(zhí)行,select
陷入阻塞
- 如果管道中有緩沖區(qū)且還可以存放至少一個數(shù)據(jù),如下:
func main() { c := make(chan string, 1) selectDemo(c) }
此時,管道可以寫入,寫操作對應(yīng)的 case 語句得到執(zhí)行,且執(zhí)行結(jié)束后函數(shù)退出。
- 如果管道有緩沖區(qū)且緩沖區(qū)中已放滿數(shù)據(jù),如下:
func main() { c := make(chan string, 1) c <- "你好,向你說再見!" selectDemo(c) }
此時,管道可以讀取,讀操作對應(yīng)的 case 語句得到執(zhí)行,且執(zhí)行結(jié)束后函數(shù)退出。
- 管道有緩沖區(qū),緩沖區(qū)中已有部分?jǐn)?shù)據(jù)還可以存入數(shù)據(jù),如下:
func main() { c := make(chan string, 2) c <- "你好,向你說再見!" selectDemo(c) }
管道的緩沖區(qū)有部分且還可以存入數(shù)據(jù),此時管道既可以讀取也可以寫入,select
將選取一個 case 語句執(zhí)行,任意一個 case 語句執(zhí)行結(jié)束后函數(shù)就退出。
總結(jié)
select 的每個 case 語句只能操作一個管道,要么寫入數(shù)據(jù),要么讀取數(shù)據(jù);
如果管道中沒有數(shù)據(jù)讀取操作則會阻塞,如果管道中沒有空余的緩沖區(qū)則寫入操作會阻塞;
當(dāng) select 的多個 case 語句中的管道均阻塞時,整個 select 語句也會陷入阻塞,直到任意一個管道解除阻塞;
如果多個 case 語句均沒有阻塞,那么 select 將隨機(jī)挑選一個 case 執(zhí)行。
以上就是Go使用select切換協(xié)程入門詳解的詳細(xì)內(nèi)容,更多關(guān)于Go select 切換協(xié)程的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
使用golang-unsafe包的注意事項(xiàng)及說明
這篇文章主要介紹了使用golang-unsafe包的注意事項(xiàng)及說明,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-02-02利用Go語言實(shí)現(xiàn)輕量級OpenLdap弱密碼檢測工具
這篇文章主要為大家詳細(xì)介紹了如何利用Go語言實(shí)現(xiàn)輕量級OpenLdap弱密碼檢測工具,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以嘗試一下2022-09-09Golang實(shí)現(xiàn)獲取與解析命令行參數(shù)
這篇文章主要為大家詳細(xì)介紹了Golang如何實(shí)現(xiàn)獲取與解析命令行參數(shù),文中的示例代碼講解詳細(xì),具有一定的借鑒價值,需要的小伙伴可以參考一下2024-01-01goland Duration 和time的區(qū)別說明
這篇文章主要介紹了goland Duration 和time的區(qū)別說明,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-12-12Go 1.22對net/http包的路由增強(qiáng)功能詳解
Go 1.22 版本對 net/http 包的路由功能進(jìn)行了增強(qiáng),引入了方法匹配(method matching)和通配符(wildcards)兩項(xiàng)新功能,本文將給大家詳細(xì)的介紹一下Go 1.22對net/http包的路由增強(qiáng)功能,需要的朋友可以參考下2024-02-02