麻豆小视频在线观看_中文黄色一级片_久久久成人精品_成片免费观看视频大全_午夜精品久久久久久久99热浪潮_成人一区二区三区四区

首頁(yè) > 編程 > Golang > 正文

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

2020-04-01 19:25:56
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友
這篇文章主要介紹了Golang極簡(jiǎn)入門教程(三):并發(fā)支持,本文講解了goroutine線程、channel 操作符等內(nèi)容,需要的朋友可以參考下
 

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

 

復(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() {
    // 開(kāi)啟一個(gè) goroutine 執(zhí)行 say 函數(shù)
    go say("world")
    say("hello")
}

 

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

 

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

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

 

使用 channel 和 goroutine 通訊能夠避免顯式使用鎖機(jī)制,通過(guò) channel 發(fā)送和接收值時(shí)默認(rèn)是阻塞的。

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

 

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

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

 

一個(gè)完整的例子:

 

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

package main
 
import "fmt"
 
// 計(jì)算數(shù)組 a 中所有元素值之和
func sum(a []int, c chan int) {
    sum := 0
    for _, v := range a {
        sum += v
    }
    // 計(jì)算結(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)
 
    // 接收兩個(gè) goroutine 發(fā)送的計(jì)算結(jié)果
    x, y := <-c, <-c
 
    fmt.Println(x, y, x+y)
}package main
 
import "fmt"
 
// 計(jì)算數(shù)組 a 中所有元素值之和
func sum(a []int, c chan int) {
    sum := 0
    for _, v := range a {
        sum += v
    }
    // 計(jì)算結(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)
 
    // 接收兩個(gè) goroutine 發(fā)送的計(jì)算結(jié)果
    x, y := <-c, <-c
 
    fmt.Println(x, y, x+y)
}

 

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

 

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

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

 

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

 

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

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

 

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

我們可以使用 for range 來(lái)接收 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)
    }
}

 

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

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

 

復(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 會(huì)在沒(méi)有任何 case 可執(zhí)行時(shí)執(zhí)行(類似于 switch):

 

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

package main
 
import (
    "fmt"
    "time"
)
 
func main() {
    // 創(chuàng)建一個(gè) tick channel
    // 在 100 毫秒后會(huì)向 tick channel 中發(fā)送當(dāng)前時(shí)間
    tick := time.Tick(100 * time.Millisecond)
    // 創(chuàng)建一個(gè) boom channel
    // 在 500 毫秒后會(huì)向 boom channel 中發(fā)送當(dāng)前時(shí)間
    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)
        }
    }
}
 

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 99精品国产一区二区三区 | 暴力肉体进入hdxxxx古装 | 91aa.app| 369看片你懂的小视频在线观看 | 九九午夜 | 国产一区二区影视 | 一级做a爰性色毛片免费 | 竹内纱里奈和大战黑人 | 精品一区二区三区免费毛片 | 国产精品一区在线观看 | 欧美黄 片免费观看 | 91 视频网站 | xxxxhd86日本护士hd | 中国大陆一级毛片 | 成人超碰97| 毛片视频网站在线观看 | 干少妇av | 悠悠成人资源亚洲一区二区 | 国产精品刺激对白麻豆99 | 欧美日韩精品不卡一区二区三区 | 国产性tv国产精品 | 国产一区二区视频在线播放 | 四季久久免费一区二区三区四区 | 久久国产在线观看 | 精品一区二区三区日本 | 中国国语毛片免费观看视频 | 国产在线精品一区二区三区不卡 | 一级黄色免费 | 久久久成人动漫 | 久久中文字幕在线观看 | 免费看a级片 | 免费色片| 超碰人人做人人爱 | 国产精品自拍啪啪 | 高清做爰免费无遮网站挡 | 免费小毛片 | 成人精品一区二区三区中文字幕 | 国产亚洲美女精品久久久2020 | 国产一级毛片在线看 | 露脸各种姿势啪啪的清纯美女 | av免费av |