婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > Go并發調用的超時處理的方法

Go并發調用的超時處理的方法

熱門標簽:廣州呼叫中心外呼系統 南通如皋申請開通400電話 西部云谷一期地圖標注 中國地圖標注省會高清 高德地圖標注口訣 浙江高速公路地圖標注 學海導航地圖標注 江西轉化率高的羿智云外呼系統 地圖標注的汽車標

之前有聊過 golang 的協程,我發覺似乎還很理論,特別是在并發安全上,所以特結合網上的一些例子,來試驗下go routine中 的 channel, select, context 的妙用。

場景-微服務調用

我們用 gin(一個web框架) 作為處理請求的工具,需求是這樣的:

一個請求 X 會去并行調用 A, B, C 三個方法,并把三個方法返回的結果加起來作為 X 請求的 Response。

但是我們這個 Response 是有時間要求的(不能超過3秒的響應時間),可能 A, B, C 中任意一個或兩個,處理邏輯十分復雜,或者數據量超大,導致處理時間超出預期,那么我們就馬上切斷,并返回已經拿到的任意個返回結果之和。

我們先來定義主函數:

func main() {
 r := gin.New()
 r.GET("/calculate", calHandler)
 http.ListenAndServe(":8008", r)
}

非常簡單,普通的請求接受和 handler 定義。其中 calHandler 是我們用來處理請求的函數。

分別定義三個假的微服務,其中第三個將會是我們超時的哪位~

func microService1() int {
 time.Sleep(1*time.Second)
 return 1
}

func microService2() int {
 time.Sleep(2*time.Second)
 return 2
}

func microService3() int {
 time.Sleep(10*time.Second)
 return 3
}

接下來,我們看看 calHandler 里到底是什么

func calHandler(c *gin.Context) {
 ...
}

要點1--并發調用

直接用 go 就好了嘛~

所以一開始我們可能就這么寫:

go microService1()
go microService2()
go microService3()

很簡單有沒有,但是等等,說好的返回值我怎么接呢?

為了能夠并行地接受處理結果,我們很容易想到用 channel 去接。

所以我們把調用服務改成這樣:

var resChan = make(chan int, 3) // 因為有3個結果,所以我們創建一個可以容納3個值的 int channel。
go func() {
 resChan - microService1()
}()

go func() {
 resChan - microService2()
}()

go func() {
 resChan - microService3()
}()

有東西接,那也要有方法去算,所以我們加一個一直循環拿 resChan 中結果并計算的方法:

var resContainer, sum int
for {
 resContainer = -resChan
 sum += resContainer
}

這樣一來我們就有一個 sum 來計算每次從 resChan 中拿出的結果了。

要點2--超時信號

還沒結束,說好的超時處理呢?

為了實現超時處理,我們需要引入一個東西,就是 context,什么是 context ?

我們這里只使用 context 的一個特性,超時通知(其實這個特性完全可以用 channel 來替代)。

可以看在定義 calHandler 的時候我們已經將 c *gin.Context 作為參數傳了進來,那我們就不用自己在聲明了。
gin.Context 簡單理解為貫穿整個 gin 聲明周期的上下文容器,有點像是分身,亦或是量子糾纏的感覺。

有了這個 gin.Context, 我們就能在一個地方對 context 做出操作,而其他正在使用 context 的函數或方法,也會感受到 context 做出的變化。

ctx, _ := context.WithTimeout(c, 3*time.Second) //定義一個超時的 context

只要時間到了,我們就能用 ctx.Done() 獲取到一個超時的 channel(通知),然后其他用到這個 ctx 的地方也會停掉,并釋放 ctx。

一般來說,ctx.Done() 是結合 select 使用的。

所以我們又需要一個循環來監聽 ctx.Done()

for {
 select {
 case - ctx.Done():
  // 返回結果
}

現在我們有兩個 for 了,是不是能夠合并下?

for {
 select {
 case resContainer = -resChan:
  sum += resContainer
  fmt.Println("add", resContainer)
 case - ctx.Done():
  fmt.Println("result:", sum)
  return
 }
}

誒嘿,看上去不錯。

不過我們怎么在正常完成微服務調用的時候輸出結果呢?

看來我們還需要一個 flag

var count int
for {
 select {
 case resContainer = -resChan:
  sum += resContainer
  count ++
  fmt.Println("add", resContainer)
  if count > 2 {
   fmt.Println("result:", sum)
   return
  }
 case - ctx.Done():
  fmt.Println("timeout result:", sum)
  return
 }
}

我們加入一個計數器,因為我們只是調用3次微服務,所以當 count 大于2的時候,我們就應該結束并輸出結果了。

要點3--并發中的等待

上面的計時器是一種偷懶的方法,因為我們知道了調用微服務的次數,如果我們并不知道,或者之后還要添加呢?
手動每次改 count 的判斷閾值會不會太沙雕了?這時候我們就要加入 sync 包了。
我們將會使用的 sync 的一個特性是 WaitGroup。它的作用是等待一組協程運行完畢后,執行接下去的步驟。

我們來改下之前微服務調用的代碼塊:

var success = make(chan int, 1) // 成功的通道標識
wg := sync.WaitGroup{} // 創建一個 waitGroup 組
wg.Add(3) // 我們往組里加3個標識,因為我們要運行3個任務
go func() {
 resChan - microService1()
 wg.Done() // 完成一個,Done()一個
}()

go func() {
 resChan - microService2()
 wg.Done()
}()

go func() {
 resChan - microService3()
 wg.Done()
}()
wg.Wait() // 直到我們前面三個標識都被 Done 了,否則程序一直會阻塞在這里
success - 1 // 我們發送一個成功信號到通道中

既然我們有了 success 這個信號,那么再把它加入到監控 for 循環中,并做些修改,刪除原來 count 判斷的部分。

go func() {
 for {
  select {
  case resContainer = -resChan:
   sum += resContainer
   fmt.Println("add", resContainer)
  case - success:
   fmt.Println("result:", sum)
   return
  case - ctx.Done():
   fmt.Println("result:", sum)
   return
  }
 }
}()

三個 case,分工明確,一個用來拿服務輸出的結果并計算,一個用來做最終的完成輸出,一個是超時輸出。
同時我們將這個循環監聽,也作為協程運行。

至此,所有的主要代碼都完成了。下面是完全版

package main

import (
 "context"
 "fmt"
 "net/http"
 "sync"
 "time"

 "github.com/gin-gonic/gin"
)

// 一個請求會觸發調用三個服務,每個服務輸出一個 int,
// 請求要求結果為三個服務輸出 int 之和
// 請求返回時間不超過3秒,大于3秒只輸出已經獲得的 int 之和
func calHandler(c *gin.Context) {
 var resContainer, sum int
 var success, resChan = make(chan int), make(chan int, 3)
 ctx, _ := context.WithTimeout(c, 3*time.Second)

 go func() {
  for {
   select {
   case resContainer = -resChan:
    sum += resContainer
    fmt.Println("add", resContainer)
   case - success:
    fmt.Println("result:", sum)
    return
   case - ctx.Done():
    fmt.Println("result:", sum)
    return
   }
  }
 }()

 wg := sync.WaitGroup{}
 wg.Add(3)
 go func() {
  resChan - microService1()
  wg.Done()
 }()

 go func() {
  resChan - microService2()
  wg.Done()
 }()

 go func() {
  resChan - microService3()
  wg.Done()
 }()
 wg.Wait()
 success - 1

 return
}

func main() {
 r := gin.New()
 r.GET("/calculate", calHandler)
 http.ListenAndServe(":8008", r)
}

func microService1() int {
 time.Sleep(1*time.Second)
 return 1
}

func microService2() int {
 time.Sleep(2*time.Second)
 return 2
}

func microService3() int {
 time.Sleep(10*time.Second)
 return 3
}

上面的程序只是簡單描述了一個調用其他微服務超時的處理場景。

實際過程中還需要加很多很多調料,才能保證接口的對外完整性。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

您可能感興趣的文章:
  • 詳解Go多協程并發環境下的錯誤處理
  • Django高并發負載均衡實現原理詳解
  • golang并發編程的實現
  • 一百行Golang代碼實現簡單并發聊天室
  • 基于Django的樂觀鎖與悲觀鎖解決訂單并發問題詳解
  • django解決訂單并發問題【推薦】
  • golang并發ping主機的方法
  • golang并發下載多個文件的方法
  • 詳解Go 并發

標簽:東營 曲靖 常州 德宏 保定 貴州 吐魯番 許昌

巨人網絡通訊聲明:本文標題《Go并發調用的超時處理的方法》,本文關鍵詞  并發,調,用的,超時,處理,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Go并發調用的超時處理的方法》相關的同類信息!
  • 本頁收集關于Go并發調用的超時處理的方法的相關信息資訊供網民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    亚洲成av人片| 亚洲精品成人少妇| 亚洲电影欧美电影有声小说| 国产精品久久久久影院亚瑟 | 在线一区二区三区四区| 中文字幕视频一区二区三区久| 国产精品久久三| 日韩一区精品视频| 激情久久五月天| 国产欧美精品在线观看| 亚洲欧洲另类国产综合| 伦理电影国产精品| 欧美性受极品xxxx喷水| 亚洲精品免费电影| 成人高清在线视频| 日韩精品一区二区三区中文不卡| 久久综合久久鬼色| 婷婷六月综合网| 91在线观看高清| 成人欧美一区二区三区白人| 国产精品主播直播| 欧美xxxxx裸体时装秀| 琪琪一区二区三区| 99久久精品免费看国产免费软件| 91精品国产欧美一区二区18| 欧美tickle裸体挠脚心vk| 亚洲一二三区视频在线观看| 国产成人精品网址| 精品美女在线观看| 日本欧美一区二区三区| 欧美一区二区三区在线电影| 亚洲欧美日韩系列| 色哟哟一区二区| 石原莉奈在线亚洲三区| 欧美午夜精品久久久久久超碰 | 久久亚洲二区三区| 不卡的av电影在线观看| 一区二区三区四区不卡在线| 日韩一区二区三区四区| 麻豆视频观看网址久久| 久久精品免视看| 欧美性色黄大片| 国产成人精品免费网站| 国产女主播一区| 欧美在线视频日韩| 成人免费观看视频| 日韩国产欧美三级| 亚洲一区二区三区小说| 国产三级欧美三级日产三级99 | 欧美日韩视频第一区| 丁香一区二区三区| 奇米精品一区二区三区四区 | 日韩精品一区二区三区四区| 国产**成人网毛片九色| 午夜视频一区二区三区| 日韩毛片高清在线播放| 欧美精品一卡两卡| 欧美日韩高清在线播放| 91成人在线精品| 国产乱码精品一品二品| 亚洲国产视频在线| 一区二区三区四区视频精品免费 | 国产一区二区三区久久久| 亚洲一区成人在线| 一区二区三区在线视频观看 | 日本一区二区三区电影| 亚洲色图视频免费播放| 日韩成人一级大片| 国产成人av资源| 国产一区二三区| 国精产品一区一区三区mba视频| 毛片不卡一区二区| 99这里只有精品| 欧美精品一区男女天堂| 中文字幕在线不卡一区二区三区| 亚洲自拍偷拍九九九| 视频一区二区中文字幕| 日韩av电影免费观看高清完整版 | 国产精品自产自拍| 91国在线观看| 久久综合色鬼综合色| 亚洲香肠在线观看| 国产成人在线视频网址| 欧美综合一区二区| 国产欧美综合在线观看第十页| 中文字幕视频一区二区三区久| 久久成人羞羞网站| 91精品国产乱码| 视频一区二区中文字幕| 高清成人免费视频| 91精品国产综合久久久久久漫画| 久久久av毛片精品| 国产一区福利在线| 2020国产成人综合网| 日本网站在线观看一区二区三区| jvid福利写真一区二区三区| 日韩美女一区二区三区| 日本女优在线视频一区二区 | 99re视频这里只有精品| 久久精品99国产精品日本| 亚洲一区二区三区在线| 亚洲男同1069视频| 国产精品视频一区二区三区不卡| 精品国产a毛片| 欧美一级电影网站| 欧美三级中文字| 色av综合在线| 色婷婷久久99综合精品jk白丝| 国产成人精品三级| 国产成人亚洲综合a∨婷婷 | 欧美少妇bbb| 精品福利在线导航| 国产大陆a不卡| 欧美成人一区二区三区在线观看 | 亚洲人成伊人成综合网小说| 亚洲欧美日韩中文播放| 丝袜美腿亚洲一区| 国产剧情一区二区| 日韩一区二区在线观看视频| 久久日韩粉嫩一区二区三区| 精品国产制服丝袜高跟| 91国在线观看| av一区二区不卡| 九色综合狠狠综合久久| 日本午夜精品视频在线观看| 亚洲专区一二三| 中文字幕一区二区视频| 精品国产一区二区亚洲人成毛片 | 91亚洲永久精品| 日日摸夜夜添夜夜添国产精品| 亚洲猫色日本管| 日韩毛片在线免费观看| 亚洲人成网站在线| 亚洲综合偷拍欧美一区色| 一区二区三区精品在线| 综合分类小说区另类春色亚洲小说欧美| 久久婷婷国产综合精品青草| 精品国产精品一区二区夜夜嗨| 欧美日本韩国一区二区三区视频| 欧美久久久久久久久| 久久久久国产精品麻豆| 国产精品久久久久久久久免费相片| 亚洲国产精品黑人久久久| 中文字幕日韩欧美一区二区三区| 最近日韩中文字幕| 石原莉奈一区二区三区在线观看| 日韩电影一二三区| 99久久综合99久久综合网站| 97久久精品人人做人人爽| 91精品在线免费| 一区二区中文视频| 国产一区二区久久| 欧美影院一区二区| 亚洲欧洲一区二区三区| 婷婷中文字幕综合| 成人av资源站| 日韩精品一区二区三区视频播放 | 亚洲欧美自拍偷拍| 国产精品一区二区黑丝| 欧美日韩久久一区| 18成人在线视频| 国产一区二区三区免费在线观看 | 日韩视频一区在线观看| 国产精品美女视频| 成人午夜碰碰视频| 欧美成人女星排行榜| 亚洲第一二三四区| 欧洲激情一区二区| 精品国产成人在线影院| 国产一区啦啦啦在线观看| 欧美最猛黑人xxxxx猛交| 国产女人水真多18毛片18精品视频| 日韩精品三区四区| 精品美女被调教视频大全网站| 琪琪久久久久日韩精品| 久久香蕉国产线看观看99| 美国欧美日韩国产在线播放| 国产午夜精品久久久久久久| 成人免费毛片嘿嘿连载视频| 国产精品麻豆欧美日韩ww| 成人黄色电影在线| 亚洲成av人片一区二区| 91精品国产欧美一区二区| 成人免费毛片片v| 亚洲第一av色| 国产免费成人在线视频| 大陆成人av片| 免费成人性网站| 专区另类欧美日韩| 日韩欧美一区二区在线视频| 韩国女主播成人在线| 亚洲精品成人精品456| 91一区二区在线观看| 亚洲免费成人av| 伊人性伊人情综合网| 午夜在线成人av| 国产盗摄精品一区二区三区在线 | 欧美成人艳星乳罩| 国产视频一区在线观看| 久久久久亚洲蜜桃|