利用go語(yǔ)言編寫(xiě)一個(gè)并發(fā)包
寫(xiě)在前面
這是一個(gè)只需要用50行代碼(核心代碼只有15行)實(shí)現(xiàn)的極其簡(jiǎn)單(原理簡(jiǎn)單、使用方法簡(jiǎn)單、功能簡(jiǎn)單)的go包mini_parallel_job,適合大部分并發(fā)任務(wù),開(kāi)箱即用。
代碼
package mini_parallel_job
import (
"fmt"
"sync"
)
type JobType func()
type JobPool interface {
AddJob(jobType JobType)
Wait()
}
type jobPool struct {
jobs []JobType
}
// 添加任務(wù)
func (j *jobPool) AddJob(job JobType) {
j.jobs = append(j.jobs, job)
}
// 開(kāi)始并且等待任務(wù)
func (j *jobPool) Wait() {
var wg sync.WaitGroup
wg.Add(len(j.jobs))
for i := range j.jobs {
jJob := j.jobs[i]
go func() {
defer func() {
wg.Done()
if err := recover(); err != nil {
fmt.Printf("err:%+v", err)
}
}()
jJob()
}()
}
wg.Wait()
}
func NewJobPool() JobPool {
return &jobPool{
jobs: make([]JobType, 0),
}
}
壓測(cè)
package mini_parallel_job
import (
"testing"
)
const (
Count = 10
)
// 并行任務(wù)
func parallelJob() {
jobPool := NewJobPool()
for i := 0; i < Count; i++ {
jobPool.AddJob(func() {
_ = fib(10)
})
}
jobPool.Wait()
}
// 串行任務(wù)
func serialJob() {
for i := 0; i < Count; i++ {
_ = fib(10)
}
}
// 任務(wù)
func fib(n int) int {
if n == 0 || n == 1 {
return n
}
return fib(n-2) + fib(n-1)
}
// 性能測(cè)試
func BenchmarkSerialJob(b *testing.B) {
for i := 0; i < b.N; i++ {
serialJob()
}
}
func BenchmarkParallelJob(b *testing.B) {
for i := 0; i < b.N; i++ {
parallelJob()
}
}
/*
BenchmarkSerialJob-12 298855 3756 ns/op
BenchmarkParallelJob-12 117189 8710 ns/op
*/
example
package main
import (
"fmt"
mini_parallel_job "mini-parallel-job"
"time"
)
const (
JobCount = 10
)
func main() {
// 串行執(zhí)行
begin1 := time.Now()
for i := 0; i < JobCount; i++ {
fib(40)
}
fmt.Println(time.Since(begin1))
// 并行執(zhí)行
begin2 := time.Now()
parallelJob := mini_parallel_job.NewJobPool()
for i := 0; i < JobCount; i++ {
parallelJob.AddJob(func() {
fib(40)
})
}
parallelJob.Wait()
fmt.Println(time.Since(begin2))
/*
結(jié)果:
7.335989407s
1.112108503s
*/
}
// 任務(wù)
func fib(n int) int {
if n == 0 || n == 1 {
return n
}
return fib(n-2) + fib(n-1)
}
總結(jié)
這段代碼僅僅實(shí)現(xiàn)使用go rountine實(shí)現(xiàn)并發(fā),sync.WaitGroup實(shí)現(xiàn)等待。
在大多數(shù)場(chǎng)景中,只需要并發(fā),并不關(guān)心并發(fā)量是多少,大多數(shù)程序員也是使用Wait函數(shù)那段代碼實(shí)現(xiàn)的(至少作者在項(xiàng)目中看到都是這樣的,并且有多處相同的代碼,基于此場(chǎng)景封裝了一下)。
如果要實(shí)現(xiàn)復(fù)雜一點(diǎn)的場(chǎng)景,比如控制最大并發(fā)量,可以稍微對(duì)上述代碼做一些修改,Wait函數(shù)中加一個(gè)指定大小的chan來(lái)控制?;蛘邊⒖甲髡吡硗庖粋€(gè)對(duì)go并發(fā)的封裝gopool,使用master-worker模式實(shí)現(xiàn)的并發(fā)控制。
到此這篇關(guān)于利用go語(yǔ)言編寫(xiě)一個(gè)并發(fā)包的文章就介紹到這了,更多相關(guān)go并發(fā)包內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
使用Go?goroutine實(shí)現(xiàn)并發(fā)的Clock服務(wù)
這篇文章主要為大家詳細(xì)介紹了如何使用Go?goroutine實(shí)現(xiàn)并發(fā)的Clock服務(wù),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-06-06
談?wù)剬?duì)Golang IO讀寫(xiě)的困惑
這篇文章主要介紹了談?wù)剬?duì)Golang IO讀寫(xiě)的困惑,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08
如何使用go實(shí)現(xiàn)創(chuàng)建WebSocket服務(wù)器
文章介紹了如何使用Go語(yǔ)言和gorilla/websocket庫(kù)創(chuàng)建一個(gè)簡(jiǎn)單的WebSocket服務(wù)器,并實(shí)現(xiàn)商品信息的實(shí)時(shí)廣播,感興趣的朋友一起看看吧2024-11-11
GoLang BoltDB數(shù)據(jù)庫(kù)詳解
這篇文章主要介紹了GoLang BoltDB數(shù)據(jù)庫(kù),boltdb是使用Go語(yǔ)言編寫(xiě)的開(kāi)源的鍵值對(duì)數(shù)據(jù)庫(kù),boltdb存儲(chǔ)數(shù)據(jù)時(shí) key和value都要求是字節(jié)數(shù)據(jù),此處需要使用到 序列化和反序列化2023-02-02
gin通過(guò)go build -tags實(shí)現(xiàn)json包切換及庫(kù)分析
這篇文章主要為大家介紹了gin通過(guò)go build -tags實(shí)現(xiàn)json包切換及庫(kù)分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-09-09
go語(yǔ)言中fallthrough的用法說(shuō)明
這篇文章主要介紹了go語(yǔ)言中fallthrough的用法說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-05-05

