Golang極簡入門教程(三):并發(fā)支持
Golang 運(yùn)行時(shí)(runtime)管理了一種輕量級線程,被叫做 goroutine。創(chuàng)建數(shù)十萬級的 goroutine 是沒有問題的。范例:
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() {
// 開啟一個(gè) goroutine 執(zhí)行 say 函數(shù)
go say("world")
say("hello")
}
我們使用 channel 和 goroutine 通訊。channel 中是一種帶有類型的通道,被用于接收和發(fā)送特定類型的值。操作符 <- 被叫做 channel 操作符(這個(gè)操作符中箭頭表明了值的流向):
// 發(fā)送 v 到 channel ch
ch <- v
// 接收 channel ch 中的值并賦值給 v
v := <-ch
使用 channel 和 goroutine 通訊能夠避免顯式使用鎖機(jī)制,通過 channel 發(fā)送和接收值時(shí)默認(rèn)是阻塞的。
通過 make 函數(shù)創(chuàng)建 channel:
// int 指定 channel 收發(fā)值的類型為 int
ch := make(chan int)
一個(gè)完整的例子:
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)來緩存被傳遞的值,向 channel 中發(fā)送時(shí)只有緩沖區(qū)滿的情況下會阻塞,接收 channel 中的值時(shí)只有在緩沖區(qū)空的情況下阻塞:
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)閉:
// 這里的 ok 為 false 表示已經(jīng)沒有值可以接收了,并且 channel 被關(guān)閉了
v, ok := <-ch
不要向已經(jīng)關(guān)閉的 channel 發(fā)送值了(will cause a panic)。
我們可以使用 for range 來接收 channel 中的值:
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。但有時(shí)候接收者必須被告知已經(jīng)沒有值可以接收了,這時(shí)候主動關(guān)閉是必要的,例如終止 for range 循環(huán)。
使用 select 語句可以讓一個(gè) goroutine 等待多個(gè)通訊操作。select 會阻塞直到某個(gè) case 能夠運(yùn)行,如果同時(shí)存在多個(gè)可執(zhí)行的,那么將隨機(jī)選擇一個(gè):
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í)行時(shí)執(zhí)行(類似于 switch):
package main
import (
"fmt"
"time"
)
func main() {
// 創(chuàng)建一個(gè) tick channel
// 在 100 毫秒后會向 tick channel 中發(fā)送當(dāng)前時(shí)間
tick := time.Tick(100 * time.Millisecond)
// 創(chuàng)建一個(gè) boom channel
// 在 500 毫秒后會向 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)
}
}
}
- 使用google-perftools優(yōu)化nginx在高并發(fā)時(shí)的性能的教程(完整版)
- Go語言并發(fā)技術(shù)詳解
- Go語言并發(fā)模型的2種編程方案
- GO語言并發(fā)編程之互斥鎖、讀寫鎖詳解
- Go語言如何并發(fā)超時(shí)處理詳解
- 如何利用Golang寫出高并發(fā)代碼詳解
- golang實(shí)現(xiàn)并發(fā)數(shù)控制的方法
- 詳解Golang 中的并發(fā)限制與超時(shí)控制
- golang中sync.Map并發(fā)創(chuàng)建、讀取問題實(shí)戰(zhàn)記錄
- Go 并發(fā)實(shí)現(xiàn)協(xié)程同步的多種解決方法
- 在Go中構(gòu)建并發(fā)TCP服務(wù)器
- Go 并發(fā)控制context實(shí)現(xiàn)原理剖析(小結(jié))
- Go并發(fā)調(diào)用的超時(shí)處理的方法
- golang 并發(fā)安全Map以及分段鎖的實(shí)現(xiàn)方法
- golang并發(fā)下載多個(gè)文件的方法
- Golang 實(shí)現(xiàn)分片讀取http超大文件流和并發(fā)控制
- golang gin 框架 異步同步 goroutine 并發(fā)操作
- Go并發(fā)4種方法簡明講解
相關(guān)文章
Go語言實(shí)現(xiàn)簡單的一個(gè)靜態(tài)WEB服務(wù)器
這篇文章主要介紹了Go語言實(shí)現(xiàn)簡單的一個(gè)靜態(tài)WEB服務(wù)器,本文給出了實(shí)現(xiàn)代碼和運(yùn)行效果,學(xué)習(xí)Golang的練手作品,需要的朋友可以參考下2014-10-10更換GORM默認(rèn)SQLite驅(qū)動出現(xiàn)的問題解決分析
這篇文章主要為大家介紹了更換GORM默認(rèn)SQLite驅(qū)動出現(xiàn)的問題解決分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-01-01Golang使用CGO與Plugin技術(shù)運(yùn)行加載C動態(tài)庫
這篇文章主要介紹了Golang使用CGO與Plugin技術(shù)運(yùn)行加載C動態(tài)庫,Golang?程序在運(yùn)行時(shí)加載C動態(tài)庫的技術(shù),跳過了Golang項(xiàng)目編譯階段需要鏈接C動態(tài)庫的過程,提高了Golang項(xiàng)目開發(fā)部署的靈活性2022-07-07使用client-go工具調(diào)用kubernetes API接口的教程詳解(v1.17版本)
這篇文章主要介紹了使用client-go工具調(diào)kubernetes API接口(v1.17版本),本文通過圖文實(shí)例相結(jié)合給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-08-08Golang實(shí)現(xiàn)Json分級解析及數(shù)字解析實(shí)踐詳解
你是否遇到過在無法準(zhǔn)確確定json層級關(guān)系的情況下對json進(jìn)行解析的需求呢?本文就來和大家介紹一次解析不確定的json對象的經(jīng)歷,以及遇到的問題和解決方法2023-02-02Go語言面向?qū)ο笾械亩鄳B(tài)你學(xué)會了嗎
面向?qū)ο笾械亩鄳B(tài)(Polymorphism)是指一個(gè)對象可以具有多種不同的形態(tài)或表現(xiàn)方式,本文將通過一些簡單的示例為大家講解一下多態(tài)的實(shí)現(xiàn),需要的可以參考下2023-07-07Go語言Http?Server框架實(shí)現(xiàn)一個(gè)簡單的httpServer
這篇文章主要為大家介紹了Go語言Http?Server框架實(shí)現(xiàn)一個(gè)簡單的httpServer抽象,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-04-04利用ChatGPT編寫一個(gè)Golang圖像壓縮函數(shù)
這篇文章主要為大家詳細(xì)介紹了如何利用ChatGPT幫我們寫了一個(gè)Golang圖像壓縮函數(shù),文中的示例代碼簡潔易懂,感興趣的小伙伴可以嘗試一下2023-04-04