golang中的select關(guān)鍵字用法總結(jié)
1.官方解釋
一個(gè)select語(yǔ)句用來(lái)選擇哪個(gè)case中的發(fā)送或接收操作可以被立即執(zhí)行。它類似于switch語(yǔ)句,但是它的case涉及到channel有關(guān)的I/O操作。即select就是用來(lái)監(jiān)聽(tīng)和channel有關(guān)的IO操作,當(dāng) IO 操作發(fā)生時(shí),觸發(fā)相應(yīng)的動(dòng)作。
2.要點(diǎn)
如果有一個(gè)或多個(gè)IO操作可以完成,則Go運(yùn)行時(shí)系統(tǒng)會(huì)隨機(jī)的選擇一個(gè)執(zhí)行,否則的話,如果有default分支,則執(zhí)行default分支語(yǔ)句,如果連default都沒(méi)有,則select語(yǔ)句會(huì)一直阻塞,直到至少有一個(gè)IO操作可以進(jìn)行
所有channel表達(dá)式都會(huì)被求值、所有被發(fā)送的表達(dá)式都會(huì)被求值。求值順序:自上而下、從左到右.
3.用法
1.使用 select 實(shí)現(xiàn) timeout 機(jī)制
timeout := make (chan bool, 1) go func() { time.Sleep(1e9) // sleep one second timeout <- true }() select { case <- timeout: fmt.Println("timeout!") }
2.使用 select 語(yǔ)句來(lái)檢測(cè) chan 是否已經(jīng)滿了
ch2 := make (chan int, 1) ch2 <- 1 select { case ch2 <- 2: default: fmt.Println("channel is full !") }
3. for-select
package main import ( "fmt" "time" ) func main() { var errChan = make(chan int) //定時(shí)2s ticker := time.NewTicker(2 * time.Second) defer ticker.Stop() go func(a chan int) { //5s發(fā)一個(gè)信號(hào) time.Sleep(time.Second * 5) errChan <- 1 }(errChan) LOOP: for { select { case <-ticker.C: { fmt.Println("Task still running") } case res, ok := <-errChan: if ok { fmt.Println("chan number:", res) break LOOP } } } fmt.Println("end!!!") } //輸出結(jié)果: //Task still running //Task still running //chan number: 1 //end!!!
附錄:
select 是 golang 中的一個(gè)控制結(jié)構(gòu),類似于 switch. 每一個(gè) case 都必須為一個(gè)通信操作,要么是發(fā)送要么是接受。
select 隨機(jī)選擇一個(gè)可運(yùn)行的 case, 如果沒(méi)有 case 可以運(yùn)行,便會(huì)阻塞,直到有 case 可以運(yùn)行。一個(gè)默認(rèn)的字句總是可以運(yùn)行的。
select { case communication clause : statement(s) case communication clause : statement(s) default : statement(s) }
以下描述 select 語(yǔ)句的語(yǔ)法
- 每個(gè) case 都必須是一個(gè)通信
- 所有 channel 表達(dá)式都會(huì)被求值
- 所有被發(fā)送的表達(dá)式都會(huì)被求值
- 如果任意某個(gè)通信可以執(zhí)行,它就會(huì)執(zhí)行;其他就會(huì)被忽略
- 如果有多個(gè) case 都可以運(yùn)行,select 會(huì)隨機(jī)公平的選出一個(gè)執(zhí)行。其他不會(huì)執(zhí)行。
否則
- 如果有 default 子句,則執(zhí)行該語(yǔ)句
- 如果沒(méi)有 default 子句,select 將阻塞,直到某個(gè)通信可以執(zhí)行;channel 或者值不會(huì)被重復(fù)求值
示例
package main import "fmt" func fibonacci(c, quit chan int) { x, y := 0, 1 for { select { case c <- x: x, y = y, x+y case <-quit: fmt.Println("quit") return } } } func main() { c := make(chan int) quit := make(chan int) // start a goroutine to print current result // no buffer in c and quit channel, so this code // would block when this goroutine try to print go func() { for i := 0; i < 10; i++ { fmt.Println(<-c) } quit <- 0 }() fibonacci(c, quit) }
總結(jié)
到此這篇關(guān)于golang中的select關(guān)鍵字用法總結(jié)的文章就介紹到這了,更多相關(guān)golang select關(guān)鍵字用法內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go?interface{}?轉(zhuǎn)切片類型的實(shí)現(xiàn)方法
本文主要介紹了Go?interface{}?轉(zhuǎn)切片類型的實(shí)現(xiàn)方法,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-02-02GO?CountMinSketch計(jì)數(shù)器(布隆過(guò)濾器思想的近似計(jì)數(shù)器)
這篇文章主要介紹了GO?CountMinSketch計(jì)數(shù)器(布隆過(guò)濾器思想的近似計(jì)數(shù)器),文章圍繞主題展開(kāi)詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的朋友可以參考一下2022-09-09go-zero源碼閱讀之布隆過(guò)濾器實(shí)現(xiàn)代碼
布隆過(guò)濾器可以用于檢索一個(gè)元素是否在一個(gè)集合中。它的優(yōu)點(diǎn)是空間效率和查詢時(shí)間都比一般的算法要好的多,缺點(diǎn)是有一定的誤識(shí)別率和刪除困難,這篇文章主要介紹了go-zero源碼閱讀-布隆過(guò)濾器,需要的朋友可以參考下2023-02-02Go語(yǔ)言協(xié)程處理數(shù)據(jù)有哪些問(wèn)題
協(xié)程(coroutine)是Go語(yǔ)言中的輕量級(jí)線程實(shí)現(xiàn),由Go運(yùn)行時(shí)(runtime)管理。本文為大家詳細(xì)介紹了Go中的協(xié)程,協(xié)程不需要搶占式調(diào)度,可以有效提高線程的任務(wù)并發(fā)性,而避免多線程的缺點(diǎn)2023-02-02