如何在 Go 中使用超時和重試來建立上下文?
php小編新一將為大家介紹如何在Go語言中使用超時和重試來建立上下文。在編寫程式碼時,我們經常會遇到需要限制某個操作的執行時間或在操作失敗時進行重試的情況。 Go語言提供了一種簡潔而強大的機制,即上下文(context),它可以幫助我們實現這些需求。透過合理地利用超時和重試機制,我們可以提高程式碼的可靠性和穩定性,同時提升使用者體驗。在接下來的文章中,我們將詳細討論如何在Go語言中使用超時和重試來建立上下文。
問題內容
我嘗試在 go 中使用超時和多次重試來建立上下文。 這是程式碼範例
func readretry(port io.readwritecloser, timeout, cnt int) []byte { fmt.println("in read retry") for i := 0; i < cnt; i++ { fmt.println("read attempt:", i) res := readwithcontext(timeout, port) if res != nil { return res } } return nil } func readwithcontext(timeout int, port io.readwritecloser) []byte { fmt.println("in readwithcontext") fmt.println("opening channel") rcvch := make(chan []byte) ctx, cancel := context.withtimeout(context.background(), time.duration(time.second*time.duration(timeout))) defer cancel() go reader(ctx, port, rcvch) for { select { case <-ctx.done(): fmt.println("reader: context cancelled") return nil case buf := <-rcvch: fmt.println("reader: got data") return buf } } } func reader(ctx context.context, port io.readwritecloser, rcvch chan []byte) { fmt.println("in reader") answ := make([]byte, 1024) buf := bytes.buffer{} var err error for { i := 0 i, err = port.read(answ) if err != nil && err != io.eof { log.printf("port.read: %v", err) } if i != 0 { answ = answ[:i] buf.write(answ) if buf.bytes()[len(buf.bytes())-1] == delimiter { fmt.print("received: ") printbuf(buf.bytes()) rcvch <- buf.bytes() //if there is no data in the first attempt, cannot write to the channel here!! return } } } }
然後,我呼叫 readretry result := readretry(port, 2, 5) // 2 秒逾時,5 次重試。但如果第一次資料還沒準備好,那麼 reader 無法寫入 rcvch 。可能已經滿了?為什麼?如果我嘗試在 readwithcontext 執行結束時關閉通道,則會發生衝突 - 寫入已關閉的通道。碰撞在哪裡?它認為,readwithcontext每次都會作為一個新實例啟動,創建一個rcvch的新實例,如果reader因超時而關閉,則所有鏈函數及其局部變數(包括通道)已被破壞。但是,看來我犯了一個錯誤。那麼,如何進行重試呢? 看看日誌是什麼樣子的:
IN READ RETRY Read attempt: 1 IN readWithContext Opening channel IN reader Start reader Received: 0x01 0x02 0x03 0x04 0x05 0x06 0x07 0x08 0x09 0x0d //<- data is received, but not written to the channel!! reader: context cancelled
解決方法
每次重試都會建立一個新的閱讀器和一個新的頻道。如果 readWithContext
超時,讀者仍然在那裡等待,並且可能最終會閱讀,但現在通道的另一端沒有人在監聽,因此讀者被洩露。
有一個 reader
goroutine 和一個通道,使用 readWithContext
從中讀取。如果上下文過期並且所有重試都用盡,您也必須停止讀取器。
以上是如何在 Go 中使用超時和重試來建立上下文?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

Go語言中使用RedisStream實現消息隊列時類型轉換問題在使用Go語言與Redis...

GoLand中自定義結構體標籤不顯示怎麼辦?在使用GoLand進行Go語言開發時,很多開發者會遇到自定義結構體標籤在�...

Go爬蟲Colly中的Queue線程問題探討在使用Go語言的Colly爬蟲庫時,開發者常常會遇到關於線程和請求隊列的問題。 �...

Go語言中字符串打印的區別:使用Println與string()函數的效果差異在Go...

Go語言中用於浮點數運算的庫介紹在Go語言(也稱為Golang)中,進行浮點數的加減乘除運算時,如何確保精度是�...

Go語言中結構體定義的兩種方式:var與type關鍵字的差異Go語言在定義結構體時,經常會看到兩種不同的寫法:一�...

Go語言中哪些庫是大公司開發或知名開源項目?在使用Go語言進行編程時,開發者常常會遇到一些常見的需求,�...
