主頁(yè) > 知識(shí)庫(kù) > 關(guān)于golang高并發(fā)的實(shí)現(xiàn)與注意事項(xiàng)說(shuō)明

關(guān)于golang高并發(fā)的實(shí)現(xiàn)與注意事項(xiàng)說(shuō)明

熱門標(biāo)簽:excel地圖標(biāo)注分布數(shù)據(jù) 涿州代理外呼系統(tǒng) 評(píng)價(jià)高的400電話辦理 電話機(jī)器人軟件免費(fèi) 外呼系統(tǒng)顯本地手機(jī)號(hào) 壽光微信地圖標(biāo)注 阿克蘇地圖標(biāo)注 百度地圖標(biāo)注后傳給手機(jī) 外呼系統(tǒng)用什么卡

一、并發(fā)的意義

并發(fā)的意義就是讓 一個(gè)程序同時(shí)做多件事情,其目的只是為了能讓程序同時(shí)做另一件事情而已,而不是為了讓程序運(yùn)行的更快(如果是多核處理器,而且任務(wù)可以分成相互獨(dú)立的部分,那么并發(fā)確實(shí)可以讓事情解決的更快)。

golang從語(yǔ)言級(jí)別上對(duì)并發(fā)提供了支持,而且在啟動(dòng)并發(fā)的方式上直接添加了語(yǔ)言級(jí)的關(guān)鍵字,不必非要按照固定的格式來(lái)定義線程函數(shù),也不必因?yàn)閱?dòng)線程的時(shí)候只能給線程函數(shù)傳遞一個(gè)參數(shù)而煩惱。

二、并發(fā)的啟動(dòng)

go的并發(fā)啟動(dòng)非常簡(jiǎn)單,幾乎沒(méi)有什么額外的準(zhǔn)備工作,要并發(fā)的函數(shù)和一般的函數(shù)沒(méi)有什么區(qū)別,參數(shù)隨意,啟動(dòng)的時(shí)候只需要加一個(gè)go關(guān)鍵之即可,其最精髓的部分在于這些協(xié)程(協(xié)程類似于線程,但是是更輕量的線程)的調(diào)度。

package main
 
import (
 "fmt"
 "time"
)
 
func comFunc() {
 fmt.Println("This is a common function.")
}
 
func main() {
 go comFunc()
 time.Sleep(time.Second * 3)
}

三、協(xié)程間的同步與通信

1、sync.WaitGroup

sync包中的WaitGroup實(shí)現(xiàn)了一個(gè)類似任務(wù)隊(duì)列的結(jié)構(gòu),你可以向隊(duì)列中加入任務(wù),任務(wù)完成后就把任務(wù)從隊(duì)列中移除,如果隊(duì)列中的任務(wù)沒(méi)有全部完成,隊(duì)列就會(huì)觸發(fā)阻塞以阻止程序繼續(xù)運(yùn)行,具體用法參考如下代碼:

package main
import (
 "fmt"
 "sync"
)
var waitGroup sync.WaitGroup
func Afunction(index int) {
 fmt.Println(index)
 waitGroup.Done() //任務(wù)完成,將任務(wù)隊(duì)列中的任務(wù)數(shù)量-1,其實(shí).Done就是.Add(-1)
}
 
func main() {
 for i := 0; i  10; i++ {
  waitGroup.Add(1) //每創(chuàng)建一個(gè)goroutine,就把任務(wù)隊(duì)列中任務(wù)的數(shù)量+1
  go Afunction(i)
 }
 waitGroup.Wait() //.Wait()這里會(huì)發(fā)生阻塞,直到隊(duì)列中所有的任務(wù)結(jié)束就會(huì)解除阻塞
}

2、channel

channel是一種golang內(nèi)置的類型,英語(yǔ)的直譯為"通道",其實(shí),它真的就是一根管道,而且是一個(gè)先進(jìn)先出的數(shù)據(jù)結(jié)構(gòu)。

我們能對(duì)channel進(jìn)行的操作只有4種:

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

(2) 放入數(shù)據(jù) (通過(guò) channel - data 操作)

(3) 取出數(shù)據(jù) (通過(guò) -channel 操作)

(4) 關(guān)閉channel (通過(guò)close()函數(shù))

channel的3種性質(zhì)入如下:

(1) channel是一種自動(dòng)阻塞的管道。

如果管道滿了,一個(gè)對(duì)channel放入數(shù)據(jù)的操作就會(huì)阻塞,直到有某個(gè)routine從channel中取出數(shù)據(jù),這個(gè)放入數(shù)據(jù)的操作才會(huì)執(zhí)行。相反同理,如果管道是空的,一個(gè)從channel取出數(shù)據(jù)的操作就會(huì)阻塞,直到某個(gè)routine向這個(gè)channel中放入數(shù)據(jù),這個(gè)取出數(shù)據(jù)的操作才會(huì)執(zhí)行。這是channel最重要的一個(gè)性質(zhì)?。?!

package main
func main() {
 ch := make(chan int, 3)
 ch - 1
 ch - 1
 ch - 1
 ch - 1 //這一行操作就會(huì)發(fā)生阻塞,因?yàn)榍叭械姆湃霐?shù)據(jù)的操作已經(jīng)把channel填滿了
}
package main
func main() {
 ch := make(chan int, 3)
 -ch //這一行會(huì)發(fā)生阻塞,因?yàn)閏hannel才剛創(chuàng)建,是空的,沒(méi)有東西可以取出
}

(2)channel分為有緩沖的channel和無(wú)緩沖的channel。

兩種channel的創(chuàng)建方法如下:

ch := make(chan int)  //無(wú)緩沖的channel,同等于make(chan int, 0)
ch := make(chan int, 5) //一個(gè)緩沖區(qū)大小為5的channel

無(wú)緩沖通道與有緩沖通道的主要區(qū)別為:無(wú)緩沖通道存取數(shù)據(jù)是同步的,即如果通道中無(wú)數(shù)據(jù),則通道一直處于阻塞狀態(tài);有緩沖通道存取數(shù)據(jù)是異步的,即存取數(shù)據(jù)互不干擾,只有當(dāng)通道中已滿時(shí),存數(shù)據(jù)操作,通道阻塞;當(dāng)通道中為空時(shí),取數(shù)據(jù)操作,通道阻塞。

因此,使用無(wú)緩沖的channel時(shí),放入操作和取出操作不能在同一個(gè)routine中,而且應(yīng)該是先確保有某個(gè)routine對(duì)它執(zhí)行取出操作,然后才能在另一個(gè)routine中執(zhí)行放入操作,否則會(huì)發(fā)生死鎖現(xiàn)象,示例如下:

package main 
import (
 "fmt"
 "sync"
)
 
var waitGroup sync.WaitGroup //使用wg等待所有routine執(zhí)行完畢,并輸出相應(yīng)的提示信息
 
func AFunc(ch chan int) {
 waitGroup.Add(1)
FLAG:
 for {
  select {
  case val := -ch:
   fmt.Println(val)
   break FLAG
  }
 }
 waitGroup.Done()
 fmt.Println("WaitGroup Done")
}
 
func main() {
 
 ch := make(chan int) //無(wú)緩沖通道
 execMode := 0        //執(zhí)行模式 0:先啟動(dòng)并發(fā),正常輸出100 1:后啟動(dòng)并發(fā),發(fā)生死鎖
 switch execMode {
 case 0:
  go AFunc(ch)
  ch - 100
 case 1:
  ch - 100
  go AFunc(ch)
 }
 waitGroup.Wait()
 close(ch)
}

使用帶緩沖的channel時(shí),因?yàn)橛芯彌_空間,所以只要緩沖區(qū)不滿,放入操作就不會(huì)阻塞,同樣,只要緩沖區(qū)不空,取出操作就不會(huì)阻塞。

而且,帶有緩沖的channel的放入和取出操作可以用在同一個(gè)routine中。

但是,一定要注意放入和取出的速率問(wèn)題,否則也會(huì)發(fā)生死鎖現(xiàn)象,示例如下:

package main
import (
 "fmt"
 "sync"
)
var waitGroup sync.WaitGroup
func AFunc(ch chan int, putMode int) {
 val := -ch
 switch putMode {
 case 0:
  fmt.Printf("Vaule=%d\n", val)
 case 1:
  fmt.Printf("Vaule=%d\n", val)
  for i := 1; i = 5; i++ {
   ch - i * val
  }
 case 2:
  fmt.Printf("Vaule=%d\n", val)
  for i := 1; i = 5; i++ {
   -ch
  }
 }
 
 waitGroup.Done()
 fmt.Println("WaitGroup Done", val)
}
 
func main() {
 ch := make(chan int, 10)
 putMode := 0 //該模式下,能夠正常輸出所有數(shù)據(jù)
 //putMode := 1//當(dāng)放入速度遠(yuǎn)大于取數(shù)速度時(shí),程序阻塞
 //putMode := 2//當(dāng)取數(shù)速度遠(yuǎn)大于放數(shù)速度時(shí),程序阻塞
 for i := 0; i  1000; i++ {
  ch - i
  waitGroup.Add(1)
  go AFunc(ch, putMode)
 }
 waitGroup.Wait()
 close(ch)
}

(3)關(guān)閉后的channel可以取數(shù)據(jù),但是不能放數(shù)據(jù)。

而且,channel在執(zhí)行了close()后并沒(méi)有真的關(guān)閉,channel中的數(shù)據(jù)全部取走之后才會(huì)真正關(guān)閉。

package main
func main() {
 ch := make(chan int, 5)
 ch - 1
 ch - 1
 close(ch)
 ch - 1 //不能對(duì)關(guān)閉的channel執(zhí)行放入操作
        
        // 會(huì)觸發(fā)panic
}
package main
func main() {
 ch := make(chan int, 5)
 ch - 1
 ch - 1
 close(ch)
 -ch //只要channel還有數(shù)據(jù),就可能執(zhí)行取出操作
 
        //正常結(jié)束
}
package main 
import "fmt" 
func main() {
 ch := make(chan int, 5)
 ch - 1
 ch - 1
 ch - 1
 ch - 1
 close(ch)  //如果執(zhí)行了close()就立即關(guān)閉channel的話,下面的循環(huán)就不會(huì)有任何輸出了
 for {
  data, ok := -ch
  if !ok {
   break
  }
  fmt.Println(data)
 }
 
 // 輸出:
 // 1
 // 1
 // 1
 // 1
 // 
 // 調(diào)用了close()后,只有channel為空時(shí),channel才會(huì)真的關(guān)閉
}

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。

您可能感興趣的文章:
  • 解決golang 關(guān)于全局變量的坑
  • 深入淺析golang zap 日志庫(kù)使用(含文件切割、分級(jí)別存儲(chǔ)和全局使用等)
  • 基于Golang 高并發(fā)問(wèn)題的解決方案
  • 使用golang編寫一個(gè)并發(fā)工作隊(duì)列
  • golang 并發(fā)編程之生產(chǎn)者消費(fèi)者詳解
  • golang 對(duì)私有函數(shù)進(jìn)行單元測(cè)試的實(shí)例
  • Golang全局變量加鎖的問(wèn)題解決

標(biāo)簽:銅川 重慶 梅河口 蘭州 雞西 吐魯番 汕頭 欽州

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《關(guān)于golang高并發(fā)的實(shí)現(xiàn)與注意事項(xiàng)說(shuō)明》,本文關(guān)鍵詞  關(guān)于,golang,高并發(fā),高,并發(fā),;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《關(guān)于golang高并發(fā)的實(shí)現(xiàn)與注意事項(xiàng)說(shuō)明》相關(guān)的同類信息!
  • 本頁(yè)收集關(guān)于關(guān)于golang高并發(fā)的實(shí)現(xiàn)與注意事項(xiàng)說(shuō)明的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章