go 簡(jiǎn)潔的并發(fā)
多核處理器越來(lái)越普及。有沒有一種簡(jiǎn)單的辦法,能夠讓我們寫的軟件釋放多核的威力?是有的。隨著Golang, Erlang, Scala等為并發(fā)設(shè)計(jì)的程序語(yǔ)言的興起,新的并發(fā)模式逐漸清晰。正如過程式編程和面向?qū)ο笠粯?,一個(gè)好的編程模式有一個(gè)極其簡(jiǎn)潔的內(nèi)核,還有在此之上豐富的外延。可以解決現(xiàn)實(shí)世界中各種各樣的問題。本文以GO語(yǔ)言為例,解釋其中內(nèi)核、外延。
前言
Java 中有一系列的線程同步的方法,go 里面有 goroutine(協(xié)程),先看下下面的代碼執(zhí)行的結(jié)果是什么呢?
package main
import (
"fmt"
)
func main() {
go func() {
fmt.Println("Goroutine 1")
}()
go func() {
fmt.Println("Goroutine 2")
}()
}
執(zhí)行以上代碼很可能看不到輸出。
因?yàn)橛锌赡苓@兩個(gè)協(xié)程還沒得到執(zhí)行,主協(xié)程就已經(jīng)結(jié)束了,而主協(xié)程結(jié)束時(shí)會(huì)結(jié)束所有其他協(xié)程,所以導(dǎo)致代碼運(yùn)行的結(jié)果什么都沒有。
估計(jì)不少新接觸 go 的童鞋都會(huì)對(duì)此郁悶😒,可能會(huì)問那么該如何等待主協(xié)程中創(chuàng)建的協(xié)程執(zhí)行完畢之后再結(jié)束主協(xié)程呢?
下面說幾種可以解決的方法:
Sleep 一段時(shí)間
在 main 方法退出之前 sleep 一段時(shí)間就可能會(huì)出現(xiàn)結(jié)果了,如下代碼:
package main
import (
"fmt"
"time"
)
func main() {
go func() {
fmt.Println("Goroutine 1")
}()
go func() {
fmt.Println("Goroutine 2")
}()
time.Sleep(time.Second * 1) // 睡眠1秒,等待上面兩個(gè)協(xié)程結(jié)束
}
這兩個(gè)簡(jiǎn)單的協(xié)程執(zhí)行消耗的時(shí)間很短的,所以你會(huì)發(fā)現(xiàn)現(xiàn)在就有結(jié)果出現(xiàn)了。
Goroutine 1
Goroutine 2
為什么上面我要說 “可能會(huì)出現(xiàn)” ?
因?yàn)?sleep 這個(gè)時(shí)間目前是設(shè)置的 1s,如果我這兩個(gè)協(xié)程里面執(zhí)行了很復(fù)雜的邏輯操作(時(shí)間大于 1s),那么就會(huì)發(fā)現(xiàn)依舊也是無(wú)結(jié)果打印出來(lái)的。
那么就可以發(fā)現(xiàn)這種方式得到問題所在了:我們無(wú)法確定需要睡眠多久
上面那種方式有問題,go 里面其實(shí)也可以用管道來(lái)實(shí)現(xiàn)同步的。
管道實(shí)現(xiàn)同步
那么用管道怎么實(shí)現(xiàn)同步呢?show code:
package main
import (
"fmt"
)
func main() {
ch := make(chan struct{})
count := 2 // count 表示活動(dòng)的協(xié)程個(gè)數(shù)
go func() {
fmt.Println("Goroutine 1")
ch - struct{}{} // 協(xié)程結(jié)束,發(fā)出信號(hào)
}()
go func() {
fmt.Println("Goroutine 2")
ch - struct{}{} // 協(xié)程結(jié)束,發(fā)出信號(hào)
}()
for range ch {
// 每次從ch中接收數(shù)據(jù),表明一個(gè)活動(dòng)的協(xié)程結(jié)束
count--
// 當(dāng)所有活動(dòng)的協(xié)程都結(jié)束時(shí),關(guān)閉管道
if count == 0 {
close(ch)
}
}
}
這種方式是一種比較完美的解決方案, goroutine / channel 它們也是在 go 里面經(jīng)常搭配在一起的一對(duì)。
sync.WaitGroup
其實(shí) go 里面也提供了更簡(jiǎn)單的方式 —— 使用 sync.WaitGroup。
WaitGroup 顧名思義,就是用來(lái)等待一組操作完成的。WaitGroup 內(nèi)部實(shí)現(xiàn)了一個(gè)計(jì)數(shù)器,用來(lái)記錄未完成的操作個(gè)數(shù),它提供了三個(gè)方法:
- Add() 用來(lái)添加計(jì)數(shù)
- Done() 用來(lái)在操作結(jié)束時(shí)調(diào)用,使計(jì)數(shù)減一
- Wait() 用來(lái)等待所有的操作結(jié)束,即計(jì)數(shù)變?yōu)?0,該函數(shù)會(huì)在計(jì)數(shù)不為 0 時(shí)等待,在計(jì)數(shù)為 0 時(shí)立即返回
繼續(xù) show code:
package main
import (
"fmt"
"sync"
)
func main() {
var wg sync.WaitGroup
wg.Add(2) // 因?yàn)橛袃蓚€(gè)動(dòng)作,所以增加2個(gè)計(jì)數(shù)
go func() {
fmt.Println("Goroutine 1")
wg.Done() // 操作完成,減少一個(gè)計(jì)數(shù)
}()
go func() {
fmt.Println("Goroutine 2")
wg.Done() // 操作完成,減少一個(gè)計(jì)數(shù)
}()
wg.Wait() // 等待,直到計(jì)數(shù)為0
}
你會(huì)發(fā)現(xiàn)也是可以看到運(yùn)行結(jié)果的,是不是發(fā)現(xiàn)這種方式是很簡(jiǎn)單的。
總結(jié)
以上所述是小編給大家介紹的Go 并發(fā)實(shí)現(xiàn)協(xié)程同步的多種解決方法,希望對(duì)大家有所幫助,如果大家有任何疑問請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
您可能感興趣的文章:- go等待一組協(xié)程結(jié)束的操作方式
- golang協(xié)程池模擬實(shí)現(xiàn)群發(fā)郵件功能
- 解決go在函數(shù)退出后子協(xié)程的退出問題
- Go使用協(xié)程交替打印字符
- Golang 之協(xié)程的用法講解
- go 協(xié)程返回值處理操作
- 淺談golang for 循環(huán)中使用協(xié)程的問題
- Go并發(fā):使用sync.WaitGroup實(shí)現(xiàn)協(xié)程同步方式