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

主頁 > 知識庫 > go語言中http超時引發的事故解決

go語言中http超時引發的事故解決

熱門標簽:百度地圖標注位置網站 如何利用高德地圖標注家 武漢百應人工智能電銷機器人 開通400電話申請流程 智能語音電銷的機器人 400手機電話免費辦理 揚州電銷外呼系統軟件 上海企業外呼系統排名 電腦外呼系統輻射大嗎

前言

我們使用的是golang標準庫的http client,對于一些http請求,我們在處理的時候,會考慮加上超時時間,防止http請求一直在請求,導致業務長時間阻塞等待。

最近同事寫了一個超時的組件,這幾天訪問量上來了,網絡也出現了波動,造成了接口在報錯超時的情況下,還是出現了請求結果的成功。

分析下具體的代碼實現

type request struct {
 method string
 url    string
 value  string
 ps     *params
}

type params struct {
 timeout     int //超時時間
 retry       int //重試次數
 headers     map[string]string
 contentType string
}

func (req *request) Do(result interface{}) ([]byte, error) {
 res, err := asyncCall(doRequest, req)
 if err != nil {
  return nil, err
 }

 if result == nil {
  return res, nil
 }

 switch req.ps.contentType {
 case "application/xml":
  if err := xml.Unmarshal(res, result); err != nil {
   return nil, err
  }
 default:
  if err := json.Unmarshal(res, result); err != nil {
   return nil, err
  }
 }

 return res, nil
}
type timeout struct {
 data []byte
 err  error
}


func doRequest(request *request) ([]byte, error) {
 var (
  req    *http.Request
  errReq error
 )
 if request.value != "null" {
  buf := strings.NewReader(request.value)
  req, errReq = http.NewRequest(request.method, request.url, buf)
  if errReq != nil {
   return nil, errReq
  }
 } else {
  req, errReq = http.NewRequest(request.method, request.url, nil)
  if errReq != nil {
   return nil, errReq
  }
 }
 // 這里的client沒有設置超時時間
 // 所以當下面檢測到一次超時的時候,會重新又發起一次請求
 // 但是老的請求其實沒有被關閉,一直在執行
 client := http.Client{}
 res, err := client.Do(req)
 ...
}

// 重試調用請求
// 當超時的時候發起一次新的請求
func asyncCall(f func(request *request) ([]byte, error), req *request) ([]byte, error) {
 p := req.ps
 ctx := context.Background()
 done := make(chan *timeout, 1)

 for i := 0; i  p.retry; i++ {
  go func(ctx context.Context) {
   // 發送HTTP請求
   res, err := f(req)
   done - timeout{
    data: res,
    err:  err,
   }
  }(ctx)
  // 錯誤主要在這里
  // 如果超時重試為3,第一次超時了,馬上又發起了一次新的請求,但是這里錯誤使用了超時的退出
  // 具體看上面
  select {
  case res := -done:
   return res.data, res.err
  case -time.After(time.Duration(p.timeout) * time.Millisecond):
  }
 }
 return nil, ecode.TimeoutErr
}

錯誤的原因

1、超時重試,之后過了一段時間沒有拿到結果就認為是超時了,但是http請求沒有被關閉;

2、錯誤使用了http的超時,具體的做法要通過context或http.client去實現,見下文;

修改之后的代碼

func doRequest(request *request) ([]byte, error) {
 var (
  req    *http.Request
  errReq error
 )
 if request.value != "null" {
  buf := strings.NewReader(request.value)
  req, errReq = http.NewRequest(request.method, request.url, buf)
  if errReq != nil {
   return nil, errReq
  }
 } else {
  req, errReq = http.NewRequest(request.method, request.url, nil)
  if errReq != nil {
   return nil, errReq
  }
 }

 // 這里通過http.Client設置超時時間
 client := http.Client{
  Timeout: time.Duration(request.ps.timeout) * time.Millisecond,
 }
 res, err := client.Do(req)
 ...
}

func asyncCall(f func(request *request) ([]byte, error), req *request) ([]byte, error) {
 p := req.ps
 // 重試的時候只有上一個http請求真的超時了,之后才會發起一次新的請求
 for i := 0; i  p.retry; i++ {
  // 發送HTTP請求
  res, err := f(req)
  // 判斷超時
  if netErr, ok := err.(net.Error); ok  netErr.Timeout() {
   continue
  }

  return res, err

 }
 return nil, ecode.TimeoutErr
}

服務設置超時

http.Server有兩個設置超時的方法:

ReadTimeout
ReadTimeout的時間計算是從連接被接受(accept)到request body完全被讀取(如果你不讀取body,那么時間截止到讀完header為止)

WriteTimeout
WriteTimeout的時間計算正常是從request header的讀取結束開始,到response write結束為止 (也就是ServeHTTP方法的生命周期)

srv := http.Server{  
    ReadTimeout: 5 * time.Second,
    WriteTimeout: 10 * time.Second,
}

 
srv.ListenAndServe()

net/http包還提供了TimeoutHandler返回了一個在給定的時間限制內運行的handler

func TimeoutHandler(h Handler, dt time.Duration, msg string) Handler

第一個參數是Handler,第二個參數是time.Duration(超時時間),第三個參數是string類型,當到達超時時間后返回的信息

func handler(w http.ResponseWriter, r *http.Request) {
 time.Sleep(3 * time.Second)
 fmt.Println("測試超時")

 w.Write([]byte("hello world"))
}

func server() {
 srv := http.Server{
  Addr:         ":8081",
  WriteTimeout: 1 * time.Second,
  Handler:      http.TimeoutHandler(http.HandlerFunc(handler), 5*time.Second, "Timeout!\n"),
 }
 if err := srv.ListenAndServe(); err != nil {
  os.Exit(1)
 }
}

客戶端設置超時

http.client
最簡單的我們通過http.Client的Timeout字段,就可以實現客戶端的超時控制

http.client超時是超時的高層實現,包含了從Dial到Response Body的整個請求流程。http.client的實現提供了一個結構體類型可以接受一個額外的time.Duration類型的Timeout屬性。這個參數定義了從請求開始到響應消息體被完全接收的時間限制。

func httpClientTimeout() {
 c := http.Client{
  Timeout: 3 * time.Second,
 }

 resp, err := c.Get("http://127.0.0.1:8081/test")
 fmt.Println(resp)
 fmt.Println(err)
}

context
net/http中的request實現了context,所以我們可以借助于context本身的超時機制,實現http中request的超時處理

func contextTimeout() {
 ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second)
 defer cancel()

 req, err := http.NewRequest("GET", "http://127.0.0.1:8081/test", nil)
 if err != nil {
  log.Fatal(err)
 }

 resp, err := http.DefaultClient.Do(req.WithContext(ctx))
 fmt.Println(resp)
 fmt.Println(err)
}

使用context的優點就是,當父context被取消時,子context就會層層退出。

http.Transport
通過Transport還可以進行一些更小維度的超時設置

  • net.Dialer.Timeout 限制建立TCP連接的時間
  • http.Transport.TLSHandshakeTimeout 限制 TLS握手的時間
  • http.Transport.ResponseHeaderTimeout 限制讀取response header的時間
  • http.Transport.ExpectContinueTimeout 限制client在發送包含 Expect: 100-continue的header到收到繼續發送body的response之間的時間等待。注意在1.6中設置這個值會禁用HTTP/2(DefaultTransport自1.6.2起是個特例)
func transportTimeout() {
 transport := http.Transport{
  DialContext:           (net.Dialer{}).DialContext,
  ResponseHeaderTimeout: 3 * time.Second,
 }

 c := http.Client{Transport: transport}

 resp, err := c.Get("http://127.0.0.1:8081/test")
 fmt.Println(resp)
 fmt.Println(err)
}

問題
如果在客戶端在超時的臨界點,觸發了超時機制,這時候服務端剛好也接收到了,http的請求

這種服務端還是可以拿到請求的數據,所以對于超時時間的設置我們需要根據實際情況進行權衡,同時我們要考慮接口的冪等性。

總結

1、所有的超時實現都是基于Deadline,Deadline是一個時間的絕對值,一旦設置他們永久生效,不管此時連接是否被使用和怎么用,所以需要每手動設置,所以如果想使用SetDeadline建立超時機制,需要每次在Read/Write操作之前調用它。

2、使用context進行超時控制的好處就是,當父context超時的時候,子context就會層層退出。

參考

【[譯]Go net/http 超時機制完全手冊】
【Go 語言 HTTP 請求超時入門】
【使用 timeout、deadline 和 context 取消參數使 Go net/http 服務更靈活】

到此這篇關于go語言中http超時引發的事故解決的文章就介紹到這了,更多相關go語言 http超時內容請搜索腳本之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • 解決Goland中利用HTTPClient發送請求超時返回EOF錯誤DEBUG
  • Go中http超時問題的排查及解決方法
  • golang http 連接超時和傳輸超時的例子

標簽:黑龍江 新余 延邊 宜賓 江西 武漢 嘉峪關 張掖

巨人網絡通訊聲明:本文標題《go語言中http超時引發的事故解決》,本文關鍵詞  語,言中,http,超時,引發,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《go語言中http超時引發的事故解決》相關的同類信息!
  • 本頁收集關于go語言中http超時引發的事故解決的相關信息資訊供網民參考!
  • 推薦文章
    成人影视在线播放| 精品视频在线观看一区二区| 色综合久久久久综合体桃花网| 麻豆系列国产剧在线观看| 欧美爱爱动态| 亚洲 男人 天堂| 精品视频在线观看一区二区| 国产不卡在线观看| 国产一区二区精品| 中文字幕一区二区三区 精品| 国产91丝袜在线播放0| 黄色福利片| 人人干人人草| 日韩免费片| 沈樵在线观看福利| 国产精品自拍在线观看| 精品视频一区二区三区免费| 久久久久久久久综合影视网| 国产麻豆精品视频| 国产精品自拍亚洲| 中文字幕Aⅴ资源网| 日日夜夜婷婷| 国产一区二区精品在线观看| 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 尤物视频网站在线观看| 国产国语在线播放视频| 精品在线免费播放| 欧美激情一区二区三区在线| 青青青草影院 | 色综合久久天天综合观看| 欧美18性精品| 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 可以免费看毛片的网站| 国产a免费观看| 天天做日日干| 欧美另类videosbestsex久久| 国产成人女人在线视频观看| 亚洲精品影院| 91麻豆精品国产自产在线| 国产极品白嫩美女在线观看看| 亚洲女人国产香蕉久久精品| 国产极品白嫩美女在线观看看| 精品在线观看一区| 国产网站免费在线观看| 99久久网站| 麻豆午夜视频| 久久99中文字幕| 精品毛片视频| 欧美激情一区二区三区视频 | 欧美一级视频免费| 欧美18性精品| 午夜欧美成人久久久久久| 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 亚洲 国产精品 日韩| 国产极品白嫩美女在线观看看| 日本在线www| 你懂的在线观看视频| 台湾美女古装一级毛片| 色综合久久天天综线观看| 一本伊大人香蕉高清在线观看| 一级片片| 国产视频一区二区三区四区| 精品视频在线观看一区二区三区| 可以免费看毛片的网站| 欧美国产日韩在线| 成人av在线播放| 成人免费福利片在线观看| 久久久久久久男人的天堂| 韩国三级视频网站| 尤物视频网站在线观看| 99热精品一区| 久草免费在线色站| 成人高清视频免费观看| 成人免费观看的视频黄页| 国产网站在线| 成人免费观看视频| 免费的黄视频| 国产视频一区在线| 色综合久久久久综合体桃花网| 在线观看导航| 国产高清视频免费| 麻豆午夜视频| 国产亚洲精品aaa大片| 国产一区二区高清视频| 欧美国产日韩一区二区三区| 麻豆系列国产剧在线观看| 日韩中文字幕一区| 色综合久久天天综线观看| 日韩avdvd| 日本免费乱理伦片在线观看2018| 国产a免费观看| 日韩免费片| 国产一区二区精品久| 欧美另类videosbestsex视频| 成人免费一级毛片在线播放视频| 黄色免费三级| 成人免费高清视频| 国产一级生活片| 一级女性全黄生活片免费| 国产一区二区精品尤物| 亚洲天堂一区二区三区四区| 天堂网中文字幕| 国产一区二区精品尤物| 日韩专区第一页| 久久精品成人一区二区三区| 国产精品12| 精品国产一区二区三区国产馆| 国产伦精品一区三区视频 | 国产a免费观看| 999久久久免费精品国产牛牛| 国产视频一区在线| 九九免费精品视频| 成人免费一级毛片在线播放视频| 国产精品免费久久| 国产一区二区精品尤物| 国产视频一区在线| 日本特黄特色aa大片免费| 91麻豆国产级在线| 亚欧成人毛片一区二区三区四区| 日韩专区一区| 免费国产在线观看| 青青青草影院 | 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 中文字幕一区二区三区 精品| 国产原创视频在线| 尤物视频网站在线| 九九免费精品视频| 亚洲天堂在线播放| 色综合久久久久综合体桃花网| 一级毛片视频播放| 久草免费在线色站| 日本在线www| 91麻豆爱豆果冻天美星空| 日韩欧美一及在线播放| 欧美激情伊人| 国产成人啪精品| 国产91精品系列在线观看| 色综合久久天天综合| 久久久成人网| 夜夜操网| 久草免费在线色站| 国产成人啪精品| 亚洲天堂免费| 国产一区二区精品在线观看| 九九久久99综合一区二区| 欧美一区二区三区性| 国产网站麻豆精品视频| 欧美a级大片| 国产成人精品影视| 中文字幕一区二区三区精彩视频 | 四虎论坛| 免费一级片在线观看| 国产视频网站在线观看| 精品视频免费在线| 欧美一区二区三区在线观看| 国产一区精品| 99久久精品国产国产毛片| 国产美女在线观看| 国产亚洲免费观看| 午夜家庭影院| 一级女性大黄生活片免费| 中文字幕Aⅴ资源网| 国产成人女人在线视频观看| 国产原创视频在线| 国产网站在线| 99久久精品费精品国产一区二区| 美国一区二区三区| 四虎影视久久| 国产高清在线精品一区a| 精品国产一区二区三区久| 午夜精品国产自在现线拍| 国产网站麻豆精品视频| 久久国产精品只做精品| 九九精品久久| 精品国产一区二区三区精东影业| 国产视频在线免费观看| 欧美一级视频免费| 免费一级片在线观看| 免费国产在线视频| 91麻豆精品国产高清在线| 国产高清在线精品一区二区| 欧美一级视| 精品国产香蕉在线播出| 国产麻豆精品高清在线播放| 国产成人精品综合在线| 亚洲天堂免费| 国产网站在线| 日韩专区亚洲综合久久| 午夜激情视频在线观看| 欧美激情一区二区三区在线播放| 国产伦久视频免费观看 视频 | 日韩在线观看免费完整版视频| 成人高清护士在线播放| 一级毛片视频播放| 高清一级做a爱过程不卡视频| 91麻豆国产福利精品| 亚欧视频在线| 99久久精品费精品国产一区二区| 国产原创视频在线| 青草国产在线|