Golang關鍵字select的常用用法總結(jié)
一、Select解決什么問題
在Golang中,兩個協(xié)程之間通信Channel(圖一),在接受協(xié)程中通過代碼表示即為<ch;如果協(xié)程需要監(jiān)聽多個Channel,只要有其中一個滿足條件,就執(zhí)行相應的邏輯(圖二),這種select的應用場景之一,代碼如下:
func TestSelect(t *testing.T) { ch1 := make(chan int, 1) ch2 := make(chan int , 1) select { case <-ch1: fmt.Println("ch1") case <-ch2: fmt.Println("ch2") default: } }
上述代碼,創(chuàng)建兩個通道,通過select監(jiān)聽協(xié)程是否有數(shù)據(jù),如果有打印相應的值,如果沒有通過default結(jié)束程序運行;
二、Select常用用法
循環(huán)阻塞監(jiān)測
func TestLoopSelect(t *testing.T) { ch1 := make(chan int, 1) ch2 := make(chan int, 1) go func() { // 每隔1s發(fā)送一條消息到channel中 for range time.Tick(1 * time.Second) { ch1 <- 1 } }() for { select { case <-ch1: fmt.Println("ch1") case <-ch2: fmt.Println("ch2") } } }
這種寫法特別常見,起一個協(xié)程,阻塞循環(huán)監(jiān)聽多個channel,如果有數(shù)據(jù)執(zhí)行對應的操作。比如說
// go-zero/core/discov/internal/registry.go func (c *cluster) watchStream(cli EtcdClient, key string, rev int64) bool { var rch clientv3.WatchChan if rev != 0 { rch = cli.Watch(clientv3.WithRequireLeader(c.context(cli)), makeKeyPrefix(key), clientv3.WithPrefix(), clientv3.WithRev(rev+1)) } else { rch = cli.Watch(clientv3.WithRequireLeader(c.context(cli)), makeKeyPrefix(key), clientv3.WithPrefix()) } for { select { case wresp, ok := <-rch: ... c.handleWatchEvents(key, wresp.Events) case <-c.done: return true } } }
上述代碼,通過for + select阻塞循環(huán)監(jiān)測注冊中心數(shù)據(jù)是否有變換,有變化的話,針對變化類型,執(zhí)行對應邏輯;
非阻塞監(jiān)控
func TestSelect(t *testing.T) { ch1 := make(chan int, 1) ch2 := make(chan int , 1) select { case <-ch1: fmt.Println("ch1") case <-ch2: fmt.Println("ch2") default: } }
這段代碼的意思是,程序執(zhí)行到select,就檢查一下channel里面是否有數(shù)據(jù),有就處理,沒有就退出;在grpc-go中,
// grpc-go/clientconn.go func DialContext(ctx context.Context, target string, opts ...DialOption) (conn *ClientConn, err error) { cc := &ClientConn{ target: target, conns: make(map[*addrConn]struct{}), dopts: defaultDialOptions(), czData: new(channelzData), } defer func() { select { case <-ctx.Done(): switch { case ctx.Err() == err: conn = nil case err == nil || !cc.dopts.returnLastError: conn, err = nil, ctx.Err() default: conn, err = nil, fmt.Errorf("%v: %v", ctx.Err(), err) } default: } }() if cc.dopts.scChan != nil { // Blocking wait for the initial service config. select { case sc, ok := <-cc.dopts.scChan: if ok { cc.sc = &sc cc.safeConfigSelector.UpdateConfigSelector(&defaultConfigSelector{&sc}) } case <-ctx.Done(): return nil, ctx.Err() } } }
上述代碼中,有兩個地方用到select,一個地方是非阻塞,檢查contex是否被取消;另外一個是阻塞等待;
三、select原理
如果想了解select的實現(xiàn),可以閱讀runtime.selectgo代碼。它主要包含三部分:
首先,檢查一下是否有準備就緒的channel(多個channel就緒,隨機選擇一個),如果有,就執(zhí)行;
其次,將當前goroutine包裝成sudog,掛載到對應的channel上;
最后,如果channel中數(shù)據(jù)準備就緒,喚醒該協(xié)程繼續(xù)執(zhí)行第一步邏輯;
【注】源碼細節(jié),感興趣并且有需求可以深入了解,但是不要陷入源碼的怪圈中;
總結(jié)
本文主要講述下面三部分內(nèi)容:
- 從Go源碼開發(fā)者的角度考慮,為什么需要select?
- 介紹了select常用的兩種寫法,一種是非阻塞的,一種是阻塞的,以及開源項目如何使用它們;
- 介紹了select的基本實現(xiàn);
以上就是Golang關鍵字select的常用用法總結(jié)的詳細內(nèi)容,更多關于go select的資料請關注腳本之家其它相關文章!
相關文章
go mod更新指定的tag的包后,go vendor內(nèi)容未更新問題
這篇文章主要介紹了go mod更新指定的tag的包后,go vendor內(nèi)容未更新問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-09-09一文帶你掌握Golang中panic與recover的使用方法
這篇文章主要介紹了Golang中panic與recover的作用和使用方法,文中的示例代碼講解詳細,具有一定的學習價值,需要的小伙伴可以參考一下2023-04-04Go語言數(shù)據(jù)結(jié)構之希爾排序示例詳解
這篇文章主要為大家介紹了Go語言數(shù)據(jù)結(jié)構之希爾排序示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-08-08