主頁 > 知識庫 > 解決golang sync.Wait()不執(zhí)行的問題

解決golang sync.Wait()不執(zhí)行的問題

熱門標簽:上海極信防封電銷卡價格 湛江crm外呼系統(tǒng)排名 重慶慶云企業(yè)400電話到哪申請 地圖標注免費定制店 寧波語音外呼系統(tǒng)公司 宿遷便宜外呼系統(tǒng)代理商 不封卡外呼系統(tǒng) 仙桃400電話辦理 鄭州智能語音電銷機器人價格

goroutine 似乎不用解釋太多,可以利用它實現(xiàn)多線程,也可以利用它來實現(xiàn)異步事件。

在使用關(guān)鍵字go的過程中,常常會將用到sync.WaitGroup,如下一段代碼。

package main
import (
	"fmt"
	"sync"
	"time"
)
func Run() {
	var wg = sync.WaitGroup{}
	go func() {
		wg.Add(1)
		fmt.Println("halo world start")
		time.Sleep(time.Second * 5)
		fmt.Println("halo world end")
		wg.Done()
	}()
	// time.Sleep(time.Millisecond * 5)
	// fmt.Println("server will start")
	wg.Wait()
}
func main() {
	Run()
}
// output:
// 

期待的結(jié)果是打印 halo world start,5秒后打印halo world end,但是結(jié)果就是什么都沒有,并且進程立即就結(jié)束了。

原因

關(guān)鍵字go是異步的,當執(zhí)行到go,不會立即執(zhí)行g(shù)o 后面的內(nèi)容,而且繼續(xù)往下執(zhí)行。此時wg.Add(1)還沒有來得及執(zhí)行,wg.Wait()就已經(jīng)執(zhí)行,即不會發(fā)生等待,進程就結(jié)束了。

怎么解決:

只需要在wg.Wait()前有其他操作,給與足夠的時間讓wg.Add(1)執(zhí)行即可,

方法一、時間等待,在wg.Wait()前加一句time.Sleep(time.Millisecond*5),既不影響性能,也能讓wg.Add(1)來得及執(zhí)行

方法二、有IO操作,在wg.Wait()有其他IO操作,比如fmt.Println("server will start"),原因是std的輸出會將進程從用戶態(tài)轉(zhuǎn)向內(nèi)核態(tài),打印命令發(fā)出后,又切回用戶態(tài),這個狀態(tài)的轉(zhuǎn)換是很有消耗的,wg.Add(1)也就有時間執(zhí)行。

Don't worry

是否有存在擔心,方法一的時間等待,等待的時候不夠長,還是讓wg.Add(1)來不及執(zhí)行。don't worry.

這里涉及到goroutine的調(diào)度問題,go進程在執(zhí)行過程中,必須從goroutine隊列中取出一個來執(zhí)行,當wg.Wait()執(zhí)行前就算執(zhí)行time.Sleep(time.Nanosecond), 一納秒,一…一…一納秒,wg.Add(1)也來得及執(zhí)行,因為主goroutine會被切換到睡眠狀態(tài),go進程必須要取一個線程來執(zhí)行,就會取到wg.Add(1)這個線程,接下來就順理成章了。

同時方法二也是異曲同工,當發(fā)出打印的事件,整個進程都會被切換到就緒態(tài),然后再被cpu執(zhí)行。

補充:【golang】sync.WaitGroup{}的wait()調(diào)用位置不同導(dǎo)致意想不到錯誤

協(xié)程go多了,總覺的天下我有,沒事就喜歡go一個協(xié)程,信手拈來,在項目中寫個如下類似代碼:

  wh := sync.WaitGroup{}
  out := make(chan string)
  go func() {
    wh.Wait()
    close(out)
  }()
  go func() {
    for i := 0; i  2; i++ {
      wh.Add(1)
      go tt(out)
      wh.Done()
    }
  }()

想著開個協(xié)程去wait所有協(xié)程組,測試一下通了,沒問題,好牛逼,協(xié)程呀??!

可多測試即便就會出現(xiàn):

send close channel

或者協(xié)程定死在某一個,還自已為是的認為自己寫的子方法估計不小心關(guān)閉了channel,找了半天只找到在wg.wait()后進行了關(guān)閉。就這樣扣了好久,還沒想到自己畫蛇添足的錯誤,經(jīng)大佬一指點,原來開一個協(xié)程,還沒等后一個協(xié)程進行wg.add(1)操作,wg.wait()就已經(jīng)過了,關(guān)閉了channel。

只好老老實實寫:

  wh := sync.WaitGroup{}
  out := make(chan string)
  go func() {
    for i := 0; i  2; i++ {
      wh.Add(1)
      go tt(out)
      wh.Done()
    }
    wh.Wait()
    close(out)
  }()

其實就是一個小小的同步問題,旁觀者清呀?。。?/p>

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

您可能感興趣的文章:
  • Golang中的sync包的WaitGroup操作
  • Golang中的sync.WaitGroup用法實例
  • 深入Golang中的sync.Pool詳解
  • golang中使用sync.Map的方法
  • golang中sync.Map并發(fā)創(chuàng)建、讀取問題實戰(zhàn)記錄
  • 在golang中使用Sync.WaitGroup解決等待的問題

標簽:物業(yè)服務(wù) 電子產(chǎn)品 儋州 西雙版納 海南 青海 遼寧 安康

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