亚洲综合原千岁中文字幕_国产精品99久久久久久久vr_无码人妻aⅴ一区二区三区浪潮_成人h动漫精品一区二区三

主頁 > 知識庫 > Go語言使用select{}阻塞main函數介紹

Go語言使用select{}阻塞main函數介紹

熱門標簽:excel地圖標注分布數據 電話機器人軟件免費 評價高的400電話辦理 壽光微信地圖標注 百度地圖標注后傳給手機 外呼系統用什么卡 涿州代理外呼系統 外呼系統顯本地手機號 阿克蘇地圖標注

很多時候我們需要讓main函數不退出,讓它在后臺一直執行,例如:

func main() {
    for i := 0; i  20; i++ { //啟動20個協程處理消息隊列中的消息
        c := consumer.New()
        go c.Start()
    }
    select {} // 阻塞
}

可能大多數人想到阻塞的方法是用channel,當然都是可以的,不過用select{}更加簡潔 :)

補充:由淺入深聊聊Golang中select的實現機制

正文

話說今天在玩select的時候發現一個問題,是這樣的:

片段1:

func main(){
 var count int
 for {
  select {
  case -time.Tick(time.Millisecond * 500):
   fmt.Println("咖啡色的羊駝")
   count++
   fmt.Println("count--->" , count)
  case -time.Tick(time.Millisecond * 499) :
   fmt.Println(time.Now().Unix())
   count++
   fmt.Println("count--->" , count)
  }
 }
}

片段2:

func main(){
 t1 := time.Tick(time.Second)
 t2 := time.Tick(time.Second)
 var count int
 for {
  select {
  case -t1:
   fmt.Println("咖啡色的羊駝")
   count++
   fmt.Println("count--->" , count)
  case -t2 :
   fmt.Println(time.Now().Unix())
   count++
   fmt.Println("count--->" , count)
  }
 }
}

兩個問題:

1.以上片段的輸出結果是?

2.如何解釋?

第一個問題好解決,跑一下就是,很明顯輸出結果肯定不同。

片段1:

1535673600
count---> 1
1535673600
count---> 2
1535673601
count---> 3

片段2:

咖啡色的羊駝
count---> 1
1535673600
count---> 2
咖啡色的羊駝
count---> 3
1535673601
count---> 4

第二個好理解,因為select監聽了兩個time的通道,所以交替出現。

那么第一個為何只有出現1個?

為了這個問題不得不把select的實現機制走一波,所以有了此文。

select機制簡述

select有這么幾個需要關注的機制

1.select+case是用于阻塞監聽goroutine的,如果沒有case,就單單一個select{},則為監聽當前程序中的goroutine,此時注意,需要有真實的goroutine在跑,否則select{}會報panic

2.select底下有多個可執行的case,則隨機執行一個。

3.select常配合for循環來監聽channel有沒有故事發生。需要注意的是在這個場景下,break只是退出當前select而不會退出for,需要用break TIP / goto的方式。

4.無緩沖的通道,則傳值后立馬close,則會在close之前阻塞,有緩沖的通道則即使close了也會繼續讓接收后面的值

5.同個通道多個goroutine進行關閉,可用recover panic的方式來判斷通道關閉問題

看完以上知識點其實還是沒法解釋本文的核心疑惑,繼續往下!

select機制詳解

select的機制可以查看/src/runtime/select.go來了解。

源碼片段解讀:

func selectgo(sel *hselect) int {
 // ...
 // case洗牌
 pollslice := slice{unsafe.Pointer(sel.pollorder), int(sel.ncase), int(sel.ncase)}
 pollorder := *(*[]uint16)(unsafe.Pointer(pollslice))
 for i := 1; i  int(sel.ncase); i++ {
  //....
 }
 // 給case排序
 lockslice := slice{unsafe.Pointer(sel.lockorder), int(sel.ncase), int(sel.ncase)}
 lockorder := *(*[]uint16)(unsafe.Pointer(lockslice))
 for i := 0; i  int(sel.ncase); i++ {
  // ...
 }
 for i := int(sel.ncase) - 1; i >= 0; i-- {
  // ...
 }
 // 加鎖該select中所有的channel
 sellock(scases, lockorder)
 // 進入loop
loop:
 // ... 
 // pass 1 - look for something already waiting
 // 按順序遍歷case來尋找可執行的case
 for i := 0; i  int(sel.ncase); i++ {
  //...
  switch cas.kind {
  case caseNil:
   continue
  case caseRecv:
   // ... goto xxx
  case caseSend:
   // ... goto xxx
  case caseDefault:
   dfli = casi
   dfl = cas
  }
 }
 // 沒有找到可以執行的case,但有default條件,這個if里就會直接退出了。
 if dfl != nil {
  // ...
 }
 // ...
 // pass 2 - enqueue on all chans
 // chan入等待隊列
 for _, casei := range lockorder {
  // ...
  switch cas.kind {
  case caseRecv:
   c.recvq.enqueue(sg)
  case caseSend:
   c.sendq.enqueue(sg)
  }
 }
 // wait for someone to wake us up
 // 等待被喚起,同時解鎖channel(selparkcommit這里實現的)
 gp.param = nil
 gopark(selparkcommit, nil, "select", traceEvGoBlockSelect, 1)
 
 // 突然有故事發生,被喚醒,再次該select下全部channel加鎖
 sellock(scases, lockorder)
 // pass 3 - dequeue from unsuccessful chans
 // 本輪最后一次循環操作,獲取可執行case,其余全部出隊列丟棄
 casi = -1
 cas = nil
 sglist = gp.waiting
 // Clear all elem before unlinking from gp.waiting.
 for sg1 := gp.waiting; sg1 != nil; sg1 = sg1.waitlink {
  sg1.isSelect = false
  sg1.elem = nil
  sg1.c = nil
 }
 gp.waiting = nil
 for _, casei := range lockorder {
  // ...
  if sg == sglist {
   // sg has already been dequeued by the G that woke us up.
   casi = int(casei)
   cas = k
  } else {
   c = k.c
   if k.kind == caseSend {
    c.sendq.dequeueSudoG(sglist)
   } else {
    c.recvq.dequeueSudoG(sglist)
   }
  }
  // ...
 }
 // 沒有的話,再走一次loop
 if cas == nil {
  goto loop
 }
 // ...
bufrecv:
 // can receive from buffer
bufsend:
 // ...
recv:
 // ...
rclose:
 // ...
send:
 // ...
retc:
 // ...
sclose:
 // send on closed channel
}

為了方便展示,專門搞了一張很丑的圖,來說明流程:

大概就是說呢,select是分四步進行的。

本文的疑惑關鍵點就在于那個loop的時候,當接收到發現一個可執行的時候,本次select不會執行的那些case對應的channel給出隊當前goroutine,就不管他們了,就丟了,由于time.Tick是現場在case里頭創建的,而不是像片段二是處于全局棧中,所以當每次任何一個執行的時候,另一個就被拋棄了,再次selelct的時候有需要重新獲取,又是新的需要重頭再來。

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

您可能感興趣的文章:
  • 詳解Golang并發操作中常見的死鎖情形
  • Go 語言中的死鎖問題解決
  • Go語言死鎖與goroutine泄露問題的解決
  • golang coroutine 的等待與死鎖用法
  • go select編譯期的優化處理邏輯使用場景分析
  • Django實現jquery select2帶搜索的下拉框
  • matplotlib之多邊形選區(PolygonSelector)的使用
  • golang中的select關鍵字用法總結
  • Go select 死鎖的一個細節

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

巨人網絡通訊聲明:本文標題《Go語言使用select{}阻塞main函數介紹》,本文關鍵詞  語言,使用,select,阻塞,main,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Go語言使用select{}阻塞main函數介紹》相關的同類信息!
  • 本頁收集關于Go語言使用select{}阻塞main函數介紹的相關信息資訊供網民參考!
  • 推薦文章
    欧美电影免费看大全| 国产成a人片在线观看视频 | 国产一区二区精品久久91| 日本在线不卡视频| 999精品视频在线| 麻豆网站在线看| 精品久久久久久中文字幕一区| 国产不卡在线看| 日韩一级精品视频在线观看| 国产高清在线精品一区二区| 午夜久久网| 韩国三级一区| 国产精品自拍亚洲| 91麻豆精品国产高清在线| 精品国产一区二区三区久久久狼| 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 成人影视在线播放| 一级毛片视频在线观看| 色综合久久天天综线观看| 一级毛片视频在线观看| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 国产亚洲免费观看| 九九九国产| 亚洲天堂在线播放| 日本特黄特色aaa大片免费| 黄色短视屏| 久久国产精品永久免费网站| 国产成a人片在线观看视频| 亚洲www美色| 欧美大片a一级毛片视频| 国产精品免费精品自在线观看| 欧美国产日韩精品| 你懂的在线观看视频| 欧美国产日韩久久久| 999久久久免费精品国产牛牛| 亚洲第一视频在线播放| 欧美18性精品| 香蕉视频亚洲一级| 精品视频免费在线| 久草免费在线视频| 国产视频久久久| 久久成人综合网| 你懂的日韩| 亚洲精品久久玖玖玖玖| 久久成人性色生活片| 日本在线播放一区| 国产a免费观看| 高清一级片| 精品久久久久久免费影院| 日日日夜夜操| 欧美激情影院| 可以免费看污视频的网站| 青青久久精品国产免费看| 成人影院久久久久久影院| 久久99中文字幕| 韩国毛片免费| 日本特黄特黄aaaaa大片| 韩国毛片免费大片| 精品久久久久久中文字幕一区| 日本在线播放一区| 国产伦久视频免费观看视频| 国产成人精品影视| 国产视频一区在线| 日韩中文字幕在线播放| 色综合久久天天综线观看| 国产伦久视频免费观看 视频| 天天做人人爱夜夜爽2020毛片| 亚飞与亚基在线观看| 日韩欧美一二三区| 成人影院久久久久久影院| 精品在线免费播放| 精品国产一区二区三区精东影业| 超级乱淫伦动漫| 天天做日日爱| 久久国产精品永久免费网站| 天天色成人网| 九九免费高清在线观看视频| 久久精品欧美一区二区| 成人a级高清视频在线观看| 日本伦理黄色大片在线观看网站| 久久99爰这里有精品国产| 天天做人人爱夜夜爽2020毛片| 欧美激情一区二区三区在线| 日韩专区一区| 中文字幕97| 一级毛片视频免费| 可以免费看污视频的网站| 欧美1区2区3区| 99色视频在线| 一级女性全黄生活片免费| 日本免费区| 尤物视频网站在线观看| 国产精品1024在线永久免费 | 美女免费精品视频在线观看| 精品久久久久久免费影院| 久久精品免视看国产明星 | 国产成人啪精品| 精品久久久久久中文| 国产91视频网| 日韩专区亚洲综合久久| 日韩一级黄色大片| 成人免费网站视频ww| 精品国产一区二区三区免费| 欧美日本免费| 午夜在线观看视频免费 成人| 欧美激情一区二区三区在线播放| 日本免费乱人伦在线观看| 999久久狠狠免费精品| 成人高清视频在线观看| 亚洲精品久久玖玖玖玖| 九九免费高清在线观看视频| 国产国产人免费视频成69堂| 亚洲爆爽| 可以免费在线看黄的网站| 美国一区二区三区| 国产麻豆精品| 欧美一级视| 国产麻豆精品免费密入口| 成人影院一区二区三区| 国产成人精品综合| 国产不卡在线观看视频| 91麻豆爱豆果冻天美星空| 青草国产在线观看| 欧美一区二区三区性| 天天色成人网| 999精品在线| 日韩在线观看视频网站| 99久久精品国产国产毛片| 色综合久久手机在线| 国产精品免费精品自在线观看| 国产91精品一区| 久久成人性色生活片| 欧美激情一区二区三区在线| 美女免费毛片| 亚欧乱色一区二区三区| 999久久久免费精品国产牛牛| 黄色福利片| 国产一区二区精品| 99久久网站| 可以免费在线看黄的网站| 精品视频在线观看一区二区| 午夜激情视频在线观看| 九九久久99| 国产麻豆精品hdvideoss| 久久精品免视看国产明星 | 毛片高清| 国产91视频网| 久久99青青久久99久久| 国产a视频| 黄视频网站在线观看| 欧美一级视频高清片| 午夜激情视频在线观看| 精品视频一区二区三区| 免费国产在线观看| 四虎影视精品永久免费网站| 深夜做爰性大片中文| 国产国语在线播放视频| 四虎久久精品国产| 国产亚洲精品成人a在线| 91麻豆国产福利精品| 日韩免费片| 成人a大片在线观看| 久久久久久久久综合影视网| 韩国三级视频网站| 亚飞与亚基在线观看| 一本高清在线| 欧美大片aaaa一级毛片| 午夜欧美成人久久久久久| 韩国三级视频网站| 免费国产在线观看| 日韩在线观看视频网站| 成人av在线播放| 国产91精品露脸国语对白| 欧美电影免费看大全| 深夜做爰性大片中文| 精品国产一区二区三区久久久蜜臀| 国产成人啪精品视频免费软件| 久草免费在线视频| 国产一区免费观看| 色综合久久天天综合绕观看| 韩国毛片免费大片| 尤物视频网站在线观看| 精品国产香蕉伊思人在线又爽又黄| 国产美女在线一区二区三区| 欧美日本国产| 999久久66久6只有精品| 天天做日日爱| 精品久久久久久中文字幕2017| a级毛片免费全部播放| 天天做人人爱夜夜爽2020| 亚洲精品中文字幕久久久久久| 青青青草影院| 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 99久久精品国产片| 午夜激情视频在线观看| 台湾美女古装一级毛片| 天天做日日爱| 日韩专区亚洲综合久久| 日日爽天天| 欧美爱色|