Go中select多路復(fù)用的實(shí)現(xiàn)示例
select 是 Go 并發(fā)編程中非常強(qiáng)大的語(yǔ)法結(jié)構(gòu),它允許程序同時(shí)等待多個(gè)通道操作的完成,從而實(shí)現(xiàn)多路復(fù)用機(jī)制,是協(xié)程調(diào)度、超時(shí)控制、通道競(jìng)爭(zhēng)等場(chǎng)景的核心工具。
一、什么是select
select 類(lèi)似于 switch 語(yǔ)句,但它用于監(jiān)聽(tīng)多個(gè)通道的發(fā)送/接收操作。一旦其中任意一個(gè)通道準(zhǔn)備就緒,select 就會(huì)執(zhí)行相應(yīng)的語(yǔ)句塊。
基本語(yǔ)法:
select {
case val := <-ch1:
// ch1 可讀時(shí)執(zhí)行
case ch2 <- data:
// ch2 可寫(xiě)時(shí)執(zhí)行
default:
// 所有通道都不準(zhǔn)備好時(shí)執(zhí)行(可選)
}
二、select 使用示例
示例1:監(jiān)聽(tīng)多個(gè)通道輸入
func main() {
ch1 := make(chan string)
ch2 := make(chan string)
go func() {
time.Sleep(1 * time.Second)
ch1 <- "消息來(lái)自 ch1"
}()
go func() {
time.Sleep(2 * time.Second)
ch2 <- "消息來(lái)自 ch2"
}()
select {
case msg1 := <-ch1:
fmt.Println(msg1)
case msg2 := <-ch2:
fmt.Println(msg2)
}
}
輸出(大概率):
消息來(lái)自 ch1
誰(shuí)先準(zhǔn)備好,誰(shuí)被執(zhí)行。
三、select的特性
- 隨機(jī)選擇:如果多個(gè)
case同時(shí)滿(mǎn)足,Go 會(huì)隨機(jī)選擇一個(gè)執(zhí)行,避免饑餓。 - 阻塞行為:當(dāng)沒(méi)有任何
case可以運(yùn)行時(shí),select會(huì)阻塞,除非有default。 - 可配合 for 使用:實(shí)現(xiàn)多路輪詢(xún)、協(xié)程調(diào)度等高級(jí)用法。
四、使用select實(shí)現(xiàn)超時(shí)機(jī)制
select 搭配 time.After() 可實(shí)現(xiàn)通道的超時(shí)控制:
select {
case msg := <-ch:
fmt.Println("收到消息:", msg)
case <-time.After(2 * time.Second):
fmt.Println("超時(shí)未收到")
}
實(shí)用場(chǎng)景:
- 網(wǎng)絡(luò)請(qǐng)求超時(shí)
- 等待任務(wù)執(zhí)行完成
- 控制并發(fā)阻塞時(shí)間
五、非阻塞通信:使用default
select {
case msg := <-ch:
fmt.Println("收到:", msg)
default:
fmt.Println("沒(méi)有收到任何數(shù)據(jù)")
}
不等待,立即返回默認(rèn)分支。
六、監(jiān)聽(tīng)通道關(guān)閉
配合 range 和 select,可以?xún)?yōu)雅處理通道關(guān)閉:
for {
select {
case msg, ok := <-ch:
if !ok {
fmt.Println("通道已關(guān)閉")
return
}
fmt.Println("收到:", msg)
}
}
七、實(shí)踐:合并多個(gè)輸入通道
func merge(ch1, ch2 <-chan string) <-chan string {
out := make(chan string)
go func() {
for {
select {
case msg := <-ch1:
out <- msg
case msg := <-ch2:
out <- msg
}
}
}()
return out
}
? 實(shí)現(xiàn)“扇入”(fan-in)模式,將多個(gè)輸入流合并成一個(gè)輸出。
八、小結(jié)
| 功能 | 是否支持 |
|---|---|
| 同時(shí)監(jiān)聽(tīng)多個(gè)通道 | ? |
| 隨機(jī)選擇就緒的通道執(zhí)行 | ? |
| 支持默認(rèn)分支防止阻塞 | ? |
| 可實(shí)現(xiàn)超時(shí)控制與輪詢(xún) | ? |
| 實(shí)現(xiàn)非阻塞收發(fā)或關(guān)閉判斷 | ? |
實(shí)戰(zhàn)建議
- 為所有關(guān)鍵的通道通信加上
select和超時(shí)控制,避免協(xié)程泄漏。 - 避免使用
select輪詢(xún)空通道導(dǎo)致死循環(huán)。 - 多通道合并、拆分時(shí),配合
select和sync.WaitGroup效果更佳。
到此這篇關(guān)于Go中select多路復(fù)用的實(shí)現(xiàn)示例的文章就介紹到這了,更多相關(guān)Go select多路復(fù)用內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go語(yǔ)言實(shí)現(xiàn)可選參數(shù)的方法小結(jié)
這篇文章主要為大家詳細(xì)介紹了Go語(yǔ)言實(shí)現(xiàn)可選參數(shù)的一些常見(jiàn)方法,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-02-02
詳解Golang中string的實(shí)現(xiàn)原理與高效使用
在Go語(yǔ)言中,無(wú)論是字符串常量、字符串變量還是代碼中出現(xiàn)的字符串字面量,它們的類(lèi)型都被統(tǒng)一設(shè)置為string,下面就跟隨小編一起來(lái)了解一下Golang中string的實(shí)現(xiàn)原理與高效使用吧2024-01-01
golang 實(shí)現(xiàn)tcp轉(zhuǎn)發(fā)代理的方法
今天小編就為大家分享一篇golang 實(shí)現(xiàn)tcp轉(zhuǎn)發(fā)代理的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-08-08
Go語(yǔ)言學(xué)習(xí)之結(jié)構(gòu)體和方法使用詳解
這篇文章主要為大家詳細(xì)介紹了Go語(yǔ)言中結(jié)構(gòu)體和方法的使用,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)Go語(yǔ)言有一定的幫助,需要的可以參考一下2022-04-04
Golang處理parquet文件實(shí)戰(zhàn)指南
這篇文章主要給大家介紹了關(guān)于Golang處理parquet文件的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Golang具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2023-03-03
Go語(yǔ)言開(kāi)發(fā)技巧必知的小細(xì)節(jié)提升效率
這篇文章主要介紹了Go語(yǔ)言開(kāi)發(fā)技巧必知的小細(xì)節(jié)提升效率,分享幾個(gè)你可能不知道的Go語(yǔ)言小細(xì)節(jié),希望能幫助大家更好地學(xué)習(xí)這門(mén)語(yǔ)言2024-01-01
Go語(yǔ)言Slice切片底層的實(shí)現(xiàn)
本文主要介紹了Go語(yǔ)言Slice切片底層的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2025-04-04

