欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

golang控制結(jié)構(gòu)select機(jī)制及使用示例詳解

 更新時(shí)間:2023年10月15日 08:56:00   作者:TimLiu  
這篇文章主要介紹了golang控制結(jié)構(gòu)select機(jī)制及使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

GO select

在 Go 語(yǔ)言中,select 是一種用于處理多個(gè)通道操作的控制結(jié)構(gòu)。它可以用于在多個(gè)通道之間進(jìn)行非阻塞的選擇操作。

select 語(yǔ)句由一系列的 case 子句組成,每個(gè) case 子句表示一個(gè)通道操作。select 語(yǔ)句會(huì)按照順序依次檢查每個(gè) case 子句,并執(zhí)行其中可執(zhí)行的操作。

select 的作用主要有以下幾個(gè)方面:

多路復(fù)用通道

select 可以同時(shí)監(jiān)聽(tīng)多個(gè)通道上的操作,一旦某個(gè)通道可讀或可寫(xiě),就會(huì)執(zhí)行相應(yīng)的操作。這樣可以避免使用阻塞的 channel 操作,提高程序的并發(fā)性能。

package main
import (
    "fmt"
    "time"
)
func main() {
    ch1 := make(chan int)
    ch2 := make(chan int)
    go func() {
        time.Sleep(2 * time.Second)
        ch1 <- 1
    }()
    go func() {
        time.Sleep(1 * time.Second)
        ch2 <- 2
    }()
    select {
    case <-ch1:
        fmt.Println("Received from ch1")
    case <-ch2:
        fmt.Println("Received from ch2")
    case <-time.After(3 * time.Second):
        fmt.Println("Timeout")
    }
}

在這個(gè)示例中,我們創(chuàng)建了兩個(gè)通道 ch1 和 ch2。然后分別在兩個(gè) goroutine 中進(jìn)行操作,通過(guò)不同的延遲時(shí)間向通道發(fā)送數(shù)據(jù)。

在 main 函數(shù)中,我們使用 select 語(yǔ)句同時(shí)監(jiān)聽(tīng) ch1 和 ch2 兩個(gè)通道,并通過(guò) <-ch1 和 <-ch2 分別接收通道中的數(shù)據(jù)。同時(shí),我們還使用 time.After 函數(shù)設(shè)置了一個(gè) 3 秒的超時(shí)時(shí)間。

在 select 語(yǔ)句的執(zhí)行過(guò)程中,會(huì)依次檢查每個(gè) case 子句。如果有多個(gè) case 子句都是可執(zhí)行的,select 會(huì)隨機(jī)選擇一個(gè)執(zhí)行。在這個(gè)示例中,由于 ch2 的數(shù)據(jù)發(fā)送時(shí)間比 ch1 早,所以最終會(huì)執(zhí)行 case <-ch2 分支,輸出 "Received from ch2"。

如果 select 語(yǔ)句中的所有通道都沒(méi)有數(shù)據(jù)可讀,并且超過(guò)了設(shè)置的超時(shí)時(shí)間,那么就會(huì)執(zhí)行 time.After 對(duì)應(yīng)的 case 分支,輸出 "Timeout"。

非阻塞的通道操作

select 語(yǔ)句中的 case 子句可以使用非阻塞的通道操作,包括發(fā)送和接收操作。如果沒(méi)有可用的通道操作,select 會(huì)立即執(zhí)行 default 子句(如果有),或者阻塞等待第一個(gè)可執(zhí)行的操作。

package main
import (
    "fmt"
)
func main() {
    ch := make(chan int, 2)
    ch <- 1 // 向通道寫(xiě)入數(shù)據(jù),此時(shí)通道未滿,操作不會(huì)被阻塞
    fmt.Println("Data written to channel")
    select {
    case ch <- 2: // 嘗試向已滿的通道再次寫(xiě)入數(shù)據(jù),由于通道已滿,操作會(huì)被立即返回
        fmt.Println("Data written to channel")
    default:
        fmt.Println("Channel is full, unable to write data")
    }
    data, ok := <-ch // 嘗試從通道讀取數(shù)據(jù),此時(shí)通道中有數(shù)據(jù),操作不會(huì)被阻塞
    if ok {
        fmt.Println("Data read from channel:", data)
    }
    select {
    case data, ok := <-ch: // 嘗試從空的通道讀取數(shù)據(jù),由于通道為空,操作會(huì)被立即返回
        if ok {
            fmt.Println("Data read from channel:", data)
        } else {
            fmt.Println("Channel is empty, unable to read data")
        }
    default:
        fmt.Println("Channel is empty, unable to read data")
    }
}

在這個(gè)示例中,我們首先創(chuàng)建了一個(gè)緩沖大小為 2 的通道 ch。然后,我們使用帶緩沖的通道進(jìn)行數(shù)據(jù)寫(xiě)入操作 ch <- 1,由于通道未滿,操作不會(huì)被阻塞。

接下來(lái),我們使用非阻塞的通道寫(xiě)入操作 ch <- 2,由于通道已滿,操作會(huì)立即返回。我們使用 select 語(yǔ)句來(lái)處理這種情況,當(dāng)無(wú)法進(jìn)行通道寫(xiě)入操作時(shí),會(huì)執(zhí)行 default 分支,輸出 "Channel is full, unable to write data"。

然后,我們嘗試從通道中讀取數(shù)據(jù) data, ok := <-ch,由于通道中有數(shù)據(jù),操作不會(huì)被阻塞。

最后,我們使用非阻塞的通道讀取操作 data, ok := <-ch,由于通道為空,操作會(huì)立即返回。同樣,我們使用 select 語(yǔ)句來(lái)處理這種情況,當(dāng)無(wú)法進(jìn)行通道讀取操作時(shí),會(huì)執(zhí)行 default 分支,輸出 "Channel is empty, unable to read data"。

超時(shí)處理

通過(guò)在 select 語(yǔ)句中結(jié)合使用 time.After 函數(shù)和通道操作,可以實(shí)現(xiàn)超時(shí)機(jī)制。例如,可以使用 select 監(jiān)聽(tīng)一個(gè)帶有超時(shí)的通道操作,當(dāng)超過(guò)指定時(shí)間時(shí),執(zhí)行相應(yīng)的操作。

package main
import (
    "fmt"
    "time"
)
func main() {
    ch := make(chan int)
    go func() {
        time.Sleep(2 * time.Second)
        ch <- 1
    }()
    select {
    case <-ch:
        fmt.Println("Received from channel")
    case <-time.After(3 * time.Second):
        fmt.Println("Timeout")
    }
}

在這個(gè)示例中,我們創(chuàng)建了一個(gè)通道 ch。然后,我們?cè)谝粋€(gè) goroutine 中進(jìn)行操作,在 2 秒后向通道發(fā)送數(shù)據(jù) ch <- 1。

在 main 函數(shù)中,我們使用 select 語(yǔ)句同時(shí)監(jiān)聽(tīng) ch 通道和 time.After 函數(shù)返回的超時(shí)通道。超時(shí)通道是一個(gè)計(jì)時(shí)器通道,在指定的時(shí)間后會(huì)發(fā)送一個(gè)值給通道。

在 select 語(yǔ)句的執(zhí)行過(guò)程中,會(huì)依次檢查每個(gè) case 子句。如果 ch 通道接收到了數(shù)據(jù),就會(huì)執(zhí)行 case <-ch 分支,輸出 "Received from channel"。如果等待時(shí)間超過(guò)了設(shè)定的超時(shí)時(shí)間(這里是 3 秒),就會(huì)執(zhí)行 time.After 對(duì)應(yīng)的 case 分支,輸出 "Timeout"。

在這個(gè)示例中,由于通道的發(fā)送操作需要 2 秒才能完成,而超時(shí)時(shí)間設(shè)定為 3 秒,所以最終會(huì)執(zhí)行 case <-ch 分支,輸出 "Received from channel"。

控制并發(fā)流程

select 可以與 goroutine 結(jié)合使用,實(shí)現(xiàn)對(duì)并發(fā)流程的控制。通過(guò)在 select 中使用通道操作來(lái)進(jìn)行同步或通信,可以協(xié)調(diào)不同 goroutine 之間的執(zhí)行順序。

package main
import (
    "fmt"
    "sync"
)
func main() {
    var wg sync.WaitGroup
    // 設(shè)置并發(fā)任務(wù)數(shù)量
    concurrency := 3
    // 創(chuàng)建一個(gè)用于控制并發(fā)的通道
    semaphore := make(chan struct{}, concurrency)
    // 假設(shè)有一組任務(wù)需要并發(fā)執(zhí)行
    tasks := []string{"task1", "task2", "task3", "task4", "task5"}
    // 遍歷任務(wù)列表
    for _, task := range tasks {
        // 增加 WaitGroup 的計(jì)數(shù)器
        wg.Add(1)
        // 啟動(dòng)一個(gè) goroutine 來(lái)執(zhí)行任務(wù)
        go func(t string) {
            // 在 goroutine 開(kāi)始前向通道發(fā)送一個(gè)信號(hào)
            semaphore <- struct{}{}
            // 執(zhí)行任務(wù)
            fmt.Println("Executing", t)
            // 模擬任務(wù)執(zhí)行時(shí)間
            // 這里可以是任何實(shí)際的任務(wù)邏輯
            // ...
            // 任務(wù)完成后從通道釋放一個(gè)信號(hào)
            <-semaphore
            // 減少 WaitGroup 的計(jì)數(shù)器
            wg.Done()
        }(task)
    }
    // 等待所有任務(wù)完成
    wg.Wait()
    fmt.Println("All tasks completed")
}

在這個(gè)示例中,我們首先定義了并發(fā)任務(wù)的數(shù)量 concurrency,這決定了同時(shí)執(zhí)行任務(wù)的最大數(shù)量。然后,我們創(chuàng)建了一個(gè)用于控制并發(fā)的通道 semaphore,通過(guò)向通道發(fā)送信號(hào)來(lái)控制并發(fā)數(shù)量。

接下來(lái),我們定義了一組需要并發(fā)執(zhí)行的任務(wù)列表 tasks。在遍歷任務(wù)列表時(shí),我們?cè)黾恿?nbsp;WaitGroup 的計(jì)數(shù)器,并啟動(dòng)一個(gè) goroutine 來(lái)執(zhí)行每個(gè)任務(wù)。

在每個(gè)任務(wù)的 goroutine 中,首先向通道 semaphore 發(fā)送一個(gè)信號(hào),以占用一個(gè)并發(fā)槽位。然后執(zhí)行任務(wù)的邏輯,這里使用了簡(jiǎn)單的輸出來(lái)表示任務(wù)的執(zhí)行。任務(wù)執(zhí)行完畢后,從通道 semaphore 中釋放一個(gè)信號(hào),以讓其他任務(wù)可以占用并發(fā)槽位。最后,減少 WaitGroup 的計(jì)數(shù)器,表示任務(wù)完成。

最后,我們使用 WaitGroup 的 Wait 方法來(lái)等待所有任務(wù)完成,確保程序在所有任務(wù)執(zhí)行完畢后再繼續(xù)執(zhí)行。

總結(jié)

以下是 select 語(yǔ)句的一些特性:

  • 如果沒(méi)有任何通道操作準(zhǔn)備好,且沒(méi)有默認(rèn)的 case 子句,那么 select 語(yǔ)句會(huì)被阻塞,直到至少有一個(gè)通道操作準(zhǔn)備好。
  • 如果有多個(gè) case 子句準(zhǔn)備好,那么會(huì)隨機(jī)選擇一個(gè)執(zhí)行。不會(huì)有優(yōu)先級(jí)或順序的保證。
  • select 語(yǔ)句可以用于發(fā)送和接收操作,也可以混合使用。
  • select 語(yǔ)句可以與 for 循環(huán)結(jié)合使用,以實(shí)現(xiàn)對(duì)多個(gè)通道的連續(xù)監(jiān)控和處理。

select 機(jī)制是 Golang 中處理并發(fā)操作的重要工具之一,它能夠很好地處理多個(gè)通道操作,避免阻塞和死鎖的問(wèn)題。

以上就是golang控制結(jié)構(gòu)select機(jī)制使用示例詳解的詳細(xì)內(nèi)容,更多關(guān)于golang select機(jī)制的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 詳解Golang如何實(shí)現(xiàn)支持隨機(jī)刪除元素的堆

    詳解Golang如何實(shí)現(xiàn)支持隨機(jī)刪除元素的堆

    堆是一種非常常用的數(shù)據(jù)結(jié)構(gòu),它能夠支持在O(1)的時(shí)間復(fù)雜度獲取到最大值(或最小值)。本文主要介紹了如何實(shí)現(xiàn)支持O(log(n))隨機(jī)刪除元素的堆,需要的可以參考一下
    2022-09-09
  • go語(yǔ)言channel實(shí)現(xiàn)多核并行化運(yùn)行的方法

    go語(yǔ)言channel實(shí)現(xiàn)多核并行化運(yùn)行的方法

    這篇文章主要介紹了go語(yǔ)言channel實(shí)現(xiàn)多核并行化運(yùn)行的方法,實(shí)例分析了channel實(shí)現(xiàn)多核并行化運(yùn)行的技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2015-03-03
  • Go并發(fā)4種方法簡(jiǎn)明講解

    Go并發(fā)4種方法簡(jiǎn)明講解

    這篇文章主要介紹了Go并發(fā)4種方法簡(jiǎn)明講解,需要的朋友可以參考下
    2022-04-04
  • golang 語(yǔ)言中錯(cuò)誤處理機(jī)制

    golang 語(yǔ)言中錯(cuò)誤處理機(jī)制

    Golang 的錯(cuò)誤處理方式可能和這些你熟悉的語(yǔ)言有所不同,今天通過(guò)本文給大家分享golang 語(yǔ)言中錯(cuò)誤處理機(jī)制,感興趣的朋友一起看看吧
    2021-08-08
  • Go并發(fā)控制Channel使用場(chǎng)景分析

    Go并發(fā)控制Channel使用場(chǎng)景分析

    使用channel來(lái)控制子協(xié)程的優(yōu)點(diǎn)是實(shí)現(xiàn)簡(jiǎn)單,缺點(diǎn)是當(dāng)需要大量創(chuàng)建協(xié)程時(shí)就需要有相同數(shù)量的channel,而且對(duì)于子協(xié)程繼續(xù)派生出來(lái)的協(xié)程不方便控制
    2021-07-07
  • go語(yǔ)言算法題解二叉樹(shù)的拷貝、鏡像和對(duì)稱

    go語(yǔ)言算法題解二叉樹(shù)的拷貝、鏡像和對(duì)稱

    這篇文章主要為大家詳細(xì)介紹了go語(yǔ)言算法題解二叉樹(shù)的拷貝、鏡像和對(duì)稱,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2023-01-01
  • go?generate代碼自動(dòng)生成指南

    go?generate代碼自動(dòng)生成指南

    這篇文章主要介紹了go?generate代碼自動(dòng)生成指南,本文將探討?go?generate?命令的使用方法、原理以及一些實(shí)際應(yīng)用場(chǎng)景,希望讀者能夠更好地理解和運(yùn)用這個(gè)強(qiáng)大的工具
    2024-01-01
  • golang中字符串和數(shù)字轉(zhuǎn)換方法

    golang中字符串和數(shù)字轉(zhuǎn)換方法

    在Golang中,可以使用strconv包中的Itoa()和Atoi()函數(shù)進(jìn)行字符串與數(shù)字之間的轉(zhuǎn)換,Itoa()用于將數(shù)字轉(zhuǎn)換為字符串,Atoi()則用于將字符串轉(zhuǎn)換回?cái)?shù)字,本文介紹golang中字符串和數(shù)字轉(zhuǎn)換方法,感興趣的朋友一起看看吧
    2024-09-09
  • GO語(yǔ)言并發(fā)之好用的sync包詳解

    GO語(yǔ)言并發(fā)之好用的sync包詳解

    標(biāo)準(zhǔn)庫(kù)中的sync包在我們的日常開(kāi)發(fā)中用的頗為廣泛,那么大家對(duì)sync包的用法知道多少呢,這篇文章就大致講一下sync包和它的使用,感興趣的可以學(xué)習(xí)一下
    2022-12-12
  • 詳解Golang如何實(shí)現(xiàn)節(jié)假日不打擾用戶

    詳解Golang如何實(shí)現(xiàn)節(jié)假日不打擾用戶

    這篇文章主要為大家介紹了Golang如何實(shí)現(xiàn)節(jié)假日不打擾用戶過(guò)程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-01-01

最新評(píng)論