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

Golang極簡入門教程(三):并發(fā)支持

 更新時間:2014年10月27日 09:50:48   投稿:junjie  
這篇文章主要介紹了Golang極簡入門教程(三):并發(fā)支持,本文講解了goroutine線程、channel 操作符等內(nèi)容,需要的朋友可以參考下

Golang 運行時(runtime)管理了一種輕量級線程,被叫做 goroutine。創(chuàng)建數(shù)十萬級的 goroutine 是沒有問題的。范例:

復(fù)制代碼 代碼如下:

package main
 
import (
    "fmt"
    "time"
)
 
func say(s string) {
    for i := 0; i < 5; i++ {
        time.Sleep(100 * time.Millisecond)
        fmt.Println(s)
    }
}
 
func main() {
    // 開啟一個 goroutine 執(zhí)行 say 函數(shù)
    go say("world")
    say("hello")
}

我們使用 channel 和 goroutine 通訊。channel 中是一種帶有類型的通道,被用于接收和發(fā)送特定類型的值。操作符 <- 被叫做 channel 操作符(這個操作符中箭頭表明了值的流向):

復(fù)制代碼 代碼如下:

// 發(fā)送 v 到 channel ch
ch <- v
// 接收 channel ch 中的值并賦值給 v
v := <-ch

使用 channel 和 goroutine 通訊能夠避免顯式使用鎖機制,通過 channel 發(fā)送和接收值時默認是阻塞的。

通過 make 函數(shù)創(chuàng)建 channel:

復(fù)制代碼 代碼如下:

// int 指定 channel 收發(fā)值的類型為 int
ch := make(chan int)

一個完整的例子:

復(fù)制代碼 代碼如下:

package main
 
import "fmt"
 
// 計算數(shù)組 a 中所有元素值之和
func sum(a []int, c chan int) {
    sum := 0
    for _, v := range a {
        sum += v
    }
    // 計算結(jié)果發(fā)送到 channel c
    c <- sum
}
 
func main() {
    a := []int{7, 2, 8, -9, 4, 0}
 
    // 創(chuàng)建 channel c
    c := make(chan int)
 
    go sum(a[:len(a)/2], c)
    go sum(a[len(a)/2:], c)
 
    // 接收兩個 goroutine 發(fā)送的計算結(jié)果
    x, y := <-c, <-c
 
    fmt.Println(x, y, x+y)
}package main
 
import "fmt"
 
// 計算數(shù)組 a 中所有元素值之和
func sum(a []int, c chan int) {
    sum := 0
    for _, v := range a {
        sum += v
    }
    // 計算結(jié)果發(fā)送到 channel c
    c <- sum
}
 
func main() {
    a := []int{7, 2, 8, -9, 4, 0}
 
    // 創(chuàng)建 channel c
    c := make(chan int)
 
    go sum(a[:len(a)/2], c)
    go sum(a[len(a)/2:], c)
 
    // 接收兩個 goroutine 發(fā)送的計算結(jié)果
    x, y := <-c, <-c
 
    fmt.Println(x, y, x+y)
}

channel 可以帶有一個緩沖區(qū)(buffer)來緩存被傳遞的值,向 channel 中發(fā)送時只有緩沖區(qū)滿的情況下會阻塞,接收 channel 中的值時只有在緩沖區(qū)空的情況下阻塞:

復(fù)制代碼 代碼如下:

package main
 
import "fmt"
 
func main() {
    // 創(chuàng)建 channel,緩沖區(qū)長度為 2
    c := make(chan int, 2)
    // 由于 channel 的緩沖區(qū)長度為 2
    // 因此發(fā)送不會阻塞
    c <- 1
    c <- 2
    fmt.Println(<-c)
    fmt.Println(<-c)
}

發(fā)送者可以調(diào)用 close 來關(guān)閉 channel,接收者可以檢測到 channel 是否被關(guān)閉:

復(fù)制代碼 代碼如下:

// 這里的 ok 為 false 表示已經(jīng)沒有值可以接收了,并且 channel 被關(guān)閉了
v, ok := <-ch

不要向已經(jīng)關(guān)閉的 channel 發(fā)送值了(will cause a panic)。

我們可以使用 for range 來接收 channel 中的值:

復(fù)制代碼 代碼如下:

package main
 
import "fmt"
 
func fibonacci(n int, c chan int) {
    x, y := 0, 1
    for i := 0; i < n; i++ {
        c <- x
        x, y = y, x+y
    }
    // 必須要關(guān)閉 c
    close(c)
}
 
func main() {
    c := make(chan int, 10)
    go fibonacci(cap(c), c)
    // 這里 for 和 range 組合使用
    // 不斷的接收 c 中的值一直到它被關(guān)閉
    for i := range c {
        fmt.Println(i)
    }
}

通常來說,我們不需要主動的關(guān)閉 channel。但有時候接收者必須被告知已經(jīng)沒有值可以接收了,這時候主動關(guān)閉是必要的,例如終止 for range 循環(huán)。

使用 select 語句可以讓一個 goroutine 等待多個通訊操作。select 會阻塞直到某個 case 能夠運行,如果同時存在多個可執(zhí)行的,那么將隨機選擇一個:

復(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)
    go func() {
        for i := 0; i < 10; i++ {
            fmt.Println(<-c)
        }
        quit <- 0
    }()
    fibonacci(c, quit)
}

select 中的 default 會在沒有任何 case 可執(zhí)行時執(zhí)行(類似于 switch):

復(fù)制代碼 代碼如下:

package main
 
import (
    "fmt"
    "time"
)
 
func main() {
    // 創(chuàng)建一個 tick channel
    // 在 100 毫秒后會向 tick channel 中發(fā)送當(dāng)前時間
    tick := time.Tick(100 * time.Millisecond)
    // 創(chuàng)建一個 boom channel
    // 在 500 毫秒后會向 boom channel 中發(fā)送當(dāng)前時間
    boom := time.After(500 * time.Millisecond)
    for {
        select {
        case <-tick:
            fmt.Println("tick.")
        case <-boom:
            fmt.Println("BOOM!")
            return
        default:
            fmt.Println("    .")
            time.Sleep(50 * time.Millisecond)
        }
    }
}

相關(guān)文章

  • Go語言實現(xiàn)簡單的一個靜態(tài)WEB服務(wù)器

    Go語言實現(xiàn)簡單的一個靜態(tài)WEB服務(wù)器

    這篇文章主要介紹了Go語言實現(xiàn)簡單的一個靜態(tài)WEB服務(wù)器,本文給出了實現(xiàn)代碼和運行效果,學(xué)習(xí)Golang的練手作品,需要的朋友可以參考下
    2014-10-10
  • 模塊一 GO語言基礎(chǔ)知識-庫源碼文件

    模塊一 GO語言基礎(chǔ)知識-庫源碼文件

    這篇文章主要介紹了模塊一 GO語言基礎(chǔ)知識-庫源碼文件,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-01-01
  • 更換GORM默認SQLite驅(qū)動出現(xiàn)的問題解決分析

    更換GORM默認SQLite驅(qū)動出現(xiàn)的問題解決分析

    這篇文章主要為大家介紹了更換GORM默認SQLite驅(qū)動出現(xiàn)的問題解決分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2024-01-01
  • Go語言中你不知道的Interface詳解

    Go語言中你不知道的Interface詳解

    對于go語言來說,設(shè)計最精妙的應(yīng)該是interface了,直白點說interface是一組method的組合。下面這篇文章主要給大家介紹了關(guān)于Go語言中你不知道的Interface的相關(guān)資料,需要的朋友可以參考借鑒,下面來一起看看吧。
    2018-02-02
  • Golang使用CGO與Plugin技術(shù)運行加載C動態(tài)庫

    Golang使用CGO與Plugin技術(shù)運行加載C動態(tài)庫

    這篇文章主要介紹了Golang使用CGO與Plugin技術(shù)運行加載C動態(tài)庫,Golang?程序在運行時加載C動態(tài)庫的技術(shù),跳過了Golang項目編譯階段需要鏈接C動態(tài)庫的過程,提高了Golang項目開發(fā)部署的靈活性
    2022-07-07
  • 使用client-go工具調(diào)用kubernetes API接口的教程詳解(v1.17版本)

    使用client-go工具調(diào)用kubernetes API接口的教程詳解(v1.17版本)

    這篇文章主要介紹了使用client-go工具調(diào)kubernetes API接口(v1.17版本),本文通過圖文實例相結(jié)合給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-08-08
  • Golang實現(xiàn)Json分級解析及數(shù)字解析實踐詳解

    Golang實現(xiàn)Json分級解析及數(shù)字解析實踐詳解

    你是否遇到過在無法準確確定json層級關(guān)系的情況下對json進行解析的需求呢?本文就來和大家介紹一次解析不確定的json對象的經(jīng)歷,以及遇到的問題和解決方法
    2023-02-02
  • Go語言面向?qū)ο笾械亩鄳B(tài)你學(xué)會了嗎

    Go語言面向?qū)ο笾械亩鄳B(tài)你學(xué)會了嗎

    面向?qū)ο笾械亩鄳B(tài)(Polymorphism)是指一個對象可以具有多種不同的形態(tài)或表現(xiàn)方式,本文將通過一些簡單的示例為大家講解一下多態(tài)的實現(xiàn),需要的可以參考下
    2023-07-07
  • Go語言Http?Server框架實現(xiàn)一個簡單的httpServer

    Go語言Http?Server框架實現(xiàn)一個簡單的httpServer

    這篇文章主要為大家介紹了Go語言Http?Server框架實現(xiàn)一個簡單的httpServer抽象,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-04-04
  • 利用ChatGPT編寫一個Golang圖像壓縮函數(shù)

    利用ChatGPT編寫一個Golang圖像壓縮函數(shù)

    這篇文章主要為大家詳細介紹了如何利用ChatGPT幫我們寫了一個Golang圖像壓縮函數(shù),文中的示例代碼簡潔易懂,感興趣的小伙伴可以嘗試一下
    2023-04-04

最新評論