Go?select使用與底層原理講解
1. select的使用
select 是 Go 提供的 IO 多路復(fù)用機(jī)制,可以用多個(gè) case 同時(shí)監(jiān)聽多個(gè) channl 的讀寫狀態(tài):
- case: 可以監(jiān)聽 channl 的讀寫信號(hào)
- default:聲明默認(rèn)操作,有該字段的 select 不會(huì)阻塞
select {
case chan <-:
// TODO
case <- chan:
// TODO
default:
// TODO
}2. 底層原理
- 每一個(gè) case 對(duì)應(yīng)的 channl 都會(huì)被封裝到一個(gè)結(jié)構(gòu)體中;
- 當(dāng)?shù)谝淮螆?zhí)行到 select 時(shí),會(huì)鎖住所有的 channl 并且,打亂 case 結(jié)構(gòu)體的順序;
- 按照打亂的順序遍歷,如果有就緒的信號(hào),就直接走對(duì)應(yīng) case 的代碼段,之后跳出 select;
- 如果沒有就緒的代碼段,但是有 default 字段,那就走 default 的代碼段,之后跳出 select;
- 如果沒有 default,那就將當(dāng)前 goroutine 加入所有 channl 的對(duì)應(yīng)等待隊(duì)列;
- 當(dāng)某一個(gè)等待隊(duì)列就緒時(shí),再次鎖住所有的 channl,遍歷一遍,將所有等待隊(duì)列中的 goroutine 取出,之后執(zhí)行就緒的代碼段,跳出select。
3. 數(shù)據(jù)結(jié)構(gòu)
每一個(gè) case 對(duì)應(yīng)的數(shù)據(jù)結(jié)構(gòu)如下:
type scase struct {
c *hchan // chan
elem unsafe.Pointer // 讀或者寫的緩沖區(qū)地址
kind uint16 //case語句的類型,是default、傳值寫數(shù)據(jù)(channel <-) 還是 取值讀數(shù)據(jù)(<- channel)
pc uintptr // race pc (for race detector / msan)
releasetime int64
}4. 幾種常見 case
學(xué)習(xí)了 select 的使用與原理,我們就能更輕松地分辨不同情況下的輸出情況了。
case 1
package main
import (
"fmt"
"time"
)
func main() {
chan1 := make(chan int)
chan2 := make(chan int)
go func() {
chan1 <- 1
time.Sleep(5 * time.Second)
}()
go func() {
chan2 <- 1
time.Sleep(5 * time.Second)
}()
select {
case <- chan1:
fmt.Println("chan1")
case <- chan2:
fmt.Println("chan2")
default:
fmt.Println("default")
}
}三種輸出都有可能。
case2
package main
import (
"fmt"
"time"
)
func main() {
chan1 := make(chan int)
chan2 := make(chan int)
select {
case <- chan1:
fmt.Println("chan1")
case <- chan2:
fmt.Println("chan2")
}
fmt.Println("main exit.")
}上述程序會(huì)一直阻塞。
case3
package main
import (
"fmt"
)
func main() {
chan1 := make(chan int)
chan2 := make(chan int)
go func() {
close(chan1)
}()
go func() {
close(chan2)
}()
select {
case <- chan1:
fmt.Println("chan1")
case <- chan2:
fmt.Println("chan2")
}
fmt.Println("main exit.")
}隨機(jī)執(zhí)行1或者2.
case4
package main
func main() {
select {
}
}對(duì)于空的 select 語句,程序會(huì)被阻塞,確切的說是當(dāng)前協(xié)程被阻塞,同時(shí) Go 自帶死鎖檢測機(jī)制,當(dāng)發(fā)現(xiàn)當(dāng)前協(xié)程再也沒有機(jī)會(huì)被喚醒時(shí),則會(huì)發(fā)生 panic。所以上述程序會(huì) panic。
到此這篇關(guān)于Go select使用與底層原理講解的文章就介紹到這了,更多相關(guān)Go select使用 內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- go語言中如何使用select的實(shí)現(xiàn)示例
- go select編譯期的優(yōu)化處理邏輯使用場景分析
- Go語言使用select{}阻塞main函數(shù)介紹
- matplotlib之多邊形選區(qū)(PolygonSelector)的使用
- Golang的select多路復(fù)用及channel使用操作
- 深入了解Go的interface{}底層原理實(shí)現(xiàn)
- Go語言CSP并發(fā)模型goroutine及channel底層實(shí)現(xiàn)原理
- Go語言中的并發(fā)goroutine底層原理
- Golang 語言map底層實(shí)現(xiàn)原理解析
相關(guān)文章
淺談Golang?Slice切片如何擴(kuò)容的實(shí)現(xiàn)
本文主要介紹了淺談Golang?Slice切片如何擴(kuò)容的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-02-02
golang http使用踩過的坑與應(yīng)對(duì)方式
這篇文章主要介紹了golang http使用踩過的坑與應(yīng)對(duì)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01
詳解如何使用go-acme/lego實(shí)現(xiàn)自動(dòng)簽發(fā)證書
這篇文章主要為大家詳細(xì)介紹了如何使用?go-acme/lego?的客戶端或庫完成證書的自動(dòng)簽發(fā),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-03-03
Go微服務(wù)網(wǎng)關(guān)的實(shí)現(xiàn)
本文主要介紹了Go微服務(wù)網(wǎng)關(guān)的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-07-07
Go語言實(shí)現(xiàn)定時(shí)器的原理及使用詳解
這篇文章主要為大家詳細(xì)介紹了Go語言實(shí)現(xiàn)定時(shí)器的兩種方法:一次性定時(shí)器(Timer)和周期性定時(shí)器(Ticker),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2022-12-12
Golang實(shí)現(xiàn)定時(shí)任務(wù)的幾種方法小結(jié)
在 Golang 開發(fā)中,定時(shí)任務(wù)是常見的需求,本文將介紹幾種在 Golang 中實(shí)現(xiàn)定時(shí)任務(wù)的方法,包括 time 包的定時(shí)器、ticker,以及第三方庫 cron,并通過示例代碼展示它們的使用方式,需要的朋友可以參考下2024-01-01
Golang監(jiān)聽日志文件并發(fā)送到kafka中
這篇文章主要介紹了Golang監(jiān)聽日志文件并發(fā)送到kafka中,日志收集項(xiàng)目的準(zhǔn)備中,本文主要講的是利用golang的tail庫,監(jiān)聽日志文件的變動(dòng),將日志信息發(fā)送到kafka中?,需要的朋友可以參考一下2022-04-04
Golang多個(gè)域名的跨域資源共享的實(shí)現(xiàn)
本文主要介紹了Golang多個(gè)域名的跨域資源共享的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2025-02-02

