詳解golang開發(fā)中select多路選擇
select 是 Golang 中的一個(gè)控制結(jié)構(gòu),語法上類似于switch 語句,只不過select是用于 goroutine 間通信的 ,每個(gè) case 必須是一個(gè)通信操作,要么是發(fā)送要么是接收,select 會(huì)隨機(jī)執(zhí)行一個(gè)可運(yùn)行的 case。如果沒有 case 可運(yùn)行,goroutine 將阻塞,直到有 case 可運(yùn)行。
select 多路選擇
select寫法上跟switch case的寫法基本一致,只不過golang的select是通信控制語句。select的執(zhí)行必須有通信的發(fā)送或者接受,如果沒有就一直阻塞。
ch := make(chan bool, 0) ch1 := make(chan bool, 0) select { case ret := <-ch: fmt.Println(ret) case ret := <-ch1: fmt.Println(ret) }
如果ch和ch1都沒有通信數(shù)據(jù)發(fā)送,select就一直阻塞,直到ch或者ch1有數(shù)據(jù)發(fā)送,select就執(zhí)行相應(yīng)的case來接受數(shù)據(jù)。
select 實(shí)現(xiàn)超時(shí)控制
我們可以利用select機(jī)制實(shí)現(xiàn)一種簡(jiǎn)單的超時(shí)控制。
先看下程序完整執(zhí)行的代碼
func service(ch chan bool) { time.Sleep(time.Second*3) ch<-true } func main() { ch := make(chan bool, 0) go service(ch) select { case ret := <-ch: fmt.Println(ret) case <-time.After(time.Second*5): fmt.Println("timeout") } } ___go_build_main_go #gosetup true
可以看到使用time.After超時(shí)定義了5S,service程序執(zhí)行3S,所以肯定沒有超時(shí),跟預(yù)想的一致。
我們?cè)倏纯闯瑫r(shí)的執(zhí)行,我們將service程序執(zhí)行時(shí)間該為6S。超時(shí)控制繼續(xù)是5S,再看下執(zhí)行效果
func service(ch chan bool) { time.Sleep(time.Second*6) ch<-true } func main() { ch := make(chan bool, 0) go service(ch) select { case ret := <-ch: fmt.Println(ret) case <-time.After(time.Second*5): fmt.Println("timeout") } } ___go_build_main_go #gosetup timeout
執(zhí)行到了超時(shí)的case,跟預(yù)想的其實(shí)是一致的。
select 判斷channel是否關(guān)閉
先看下接受數(shù)據(jù)的語法
val,ok <- ch ok true 正常接收數(shù)據(jù) ok false 通道關(guān)閉
可以看到接受數(shù)據(jù)其實(shí)有兩個(gè)參數(shù),第二個(gè)bool值會(huì)反應(yīng)channel是否關(guān)閉,是否可以正常接受數(shù)據(jù)。
看下測(cè)試代碼
我們寫了一個(gè)數(shù)據(jù)發(fā)送者,兩個(gè)數(shù)據(jù)接收者,當(dāng)發(fā)送者關(guān)閉channel的時(shí)候,兩個(gè)接收者的 goroutine 可以通過以上的語法判斷channel是否關(guān)閉,決定自己的 goroutine 是否結(jié)束。
func sender(ch chan int, wg *sync.WaitGroup) { for i:=0;i<10;i++ { ch<-i } close(ch) wg.Done() } func receiver(ch chan int, wg *sync.WaitGroup) { for { if val,ok := <-ch;ok { fmt.Println(fmt.Sprintf("%d,%s",val, "revevier")) } else { fmt.Println("quit recevier") break; } } wg.Done() } func receiver2(ch chan int, wg *sync.WaitGroup) { for { if val,ok := <-ch;ok { fmt.Println(fmt.Sprintf("%d,%s",val, "revevier2")) } else { fmt.Println("quit recevier2") break; } } wg.Done() } func main() { ch := make(chan int, 0) wg := &sync.WaitGroup{} wg.Add(1) go sender(ch, wg) wg.Add(1) go receiver(ch, wg) wg.Add(1) go receiver2(ch, wg) wg.Wait() }
執(zhí)行結(jié)果
0,revevier2
2,revevier2
3,revevier2
4,revevier2
5,revevier2
6,revevier2
7,revevier2
1,revevier
9,revevier
quit recevier
8,revevier2
quit recevier2
可以看到一個(gè)數(shù)據(jù)發(fā)送者,兩個(gè)數(shù)據(jù)接收者,當(dāng)channel關(guān)閉的時(shí)候,兩個(gè)數(shù)據(jù)接收者都收到了channel關(guān)閉的通知。
需要注意的是,給一個(gè)已經(jīng)關(guān)閉的channel發(fā)送數(shù)據(jù),程序會(huì)panic,從一個(gè)已經(jīng)關(guān)閉的channel接收數(shù)據(jù),會(huì)接收到?jīng)]有參考意義的channel類型的0值數(shù)據(jù),Int是0,string是空...
select 退出計(jì)時(shí)器等程序
開發(fā)中經(jīng)常會(huì)經(jīng)常會(huì)使用輪訓(xùn)計(jì)時(shí)器,但是當(dāng)程序退出時(shí),輪訓(xùn)計(jì)時(shí)器無法關(guān)閉的問題。其實(shí)select是可以解決這個(gè)問題的。
如果我們有一個(gè)輪訓(xùn)任務(wù),需要一個(gè)timer,每隔3S執(zhí)行邏輯,過完10S之后關(guān)閉這個(gè)timer。
看下代碼
func TimeTick(wg *sync.WaitGroup,q chan bool) { defer wg.Done() t := time.NewTicker(time.Second*3) defer t.Stop() for { select { case <-q: fmt.Println("quit") return case <-t.C: fmt.Println("seconds timer") } } } func main() { q := make(chan bool) wg := new(sync.WaitGroup) wg.Add(1) go TimeTick(wg,q) time.Sleep(time.Second*10) close(q) wg.Wait() }
執(zhí)行結(jié)果
seconds timer
seconds timer
seconds timer
quit
很優(yōu)雅的通過關(guān)閉channel退出了輪訓(xùn)計(jì)時(shí)器 goroutine,
到此這篇關(guān)于golang開發(fā)中select多路選擇的文章就介紹到這了,更多相關(guān)golang select多路選擇內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go語言模擬while語句實(shí)現(xiàn)無限循環(huán)的方法
這篇文章主要介紹了Go語言模擬while語句實(shí)現(xiàn)無限循環(huán)的方法,實(shí)例分析了for語句模擬while語句的技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-02-02?Go?語言實(shí)現(xiàn)?HTTP?文件上傳和下載
這篇文章主要介紹了Go語言實(shí)現(xiàn)HTTP文件上傳和下載,文章圍繞主題展開詳細(xì)的內(nèi)容戒殺,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-09-09Golang實(shí)現(xiàn)多存儲(chǔ)驅(qū)動(dòng)設(shè)計(jì)SDK案例
這篇文章主要介紹了Golang實(shí)現(xiàn)多存儲(chǔ)驅(qū)動(dòng)設(shè)計(jì)SDK案例,Gocache是一個(gè)基于Go語言編寫的多存儲(chǔ)驅(qū)動(dòng)的緩存擴(kuò)展組件,更多具體內(nèi)容感興趣的小伙伴可以參考一下2022-09-09詳解如何通過Go來操作Redis實(shí)現(xiàn)簡(jiǎn)單的讀寫操作
作為最常用的分布式緩存中間件——Redis,了解運(yùn)作原理和如何使用是十分有必要的,今天來學(xué)習(xí)如何通過Go來操作Redis實(shí)現(xiàn)基本的讀寫操作,需要的朋友可以參考下2023-09-09Golang實(shí)現(xiàn)短網(wǎng)址/短鏈服務(wù)的開發(fā)筆記分享
這篇文章主要為大家詳細(xì)介紹了如何使用Golang實(shí)現(xiàn)短網(wǎng)址/短鏈服務(wù),文中的示例代碼講解詳細(xì),具有一定的學(xué)習(xí)價(jià)值,感興趣的小伙伴可以了解一下2023-05-05Golang中文字符串截取函數(shù)實(shí)現(xiàn)原理
在golang中可以通過切片截取一個(gè)數(shù)組或字符串,但是當(dāng)截取的字符串是中文時(shí),可能會(huì)出現(xiàn)問題,下面我們來自定義個(gè)函數(shù)解決Golang中文字符串截取問題2018-03-03