GoLang之使用Context控制請(qǐng)求超時(shí)的實(shí)現(xiàn)
起因
之前接觸了一個(gè)需求:提供一個(gè)接口,這個(gè)接口有一個(gè)超時(shí)時(shí)間,如果超時(shí)了返回超時(shí)異常;這個(gè)接口中調(diào)用其他的接口,如果調(diào)用超時(shí)了,所有請(qǐng)求全部結(jié)束。
在這個(gè)接口中,我使用了go協(xié)程去調(diào)用其他接口,所以不僅涉及到請(qǐng)求的超時(shí)控制,而且還涉及到父協(xié)程對(duì)子協(xié)程的控制問(wèn)題。在翻閱了一些資料之后,了解到了Context的基本知識(shí)。
Context
Context是golang.org.pkg下的一個(gè)包,類型是接口類型。主要功能有
父協(xié)程控制所有的子協(xié)程
Context可以通過(guò)context.Background()
或者 context.TODO()
創(chuàng)建一個(gè)空的context。兩個(gè)區(qū)別在于TODO
context可以進(jìn)行派生,創(chuàng)建出子context。context有四種不同的子context:
(1)WithCancel
:方法入?yún)⑹且粋€(gè)context;返回值是一個(gè)帶有新Done的父context的副本,以及cancel函數(shù)。當(dāng)調(diào)用cancel函數(shù)時(shí),通道將被關(guān)閉。關(guān)閉規(guī)則:會(huì)先關(guān)閉內(nèi)部的接收通道;通道關(guān)閉了接收該通道的操作會(huì)立即返回(即done返回的通道),并且context會(huì)向它所有的子值傳遞信號(hào),如果子context還有子context,那這個(gè)撤銷信號(hào)就會(huì)一級(jí)一級(jí)傳遞下去。最后這個(gè)context會(huì)斷開(kāi)其與父context的連接。
(2)WithDeadline
和WithTimeout
(本次問(wèn)題解決就使用的是這個(gè)):WithDeadline或者WithTimeout的功能極為相似。都是返回可以被撤銷的Context子值。它們不但可以被手動(dòng)撤銷,還會(huì)依據(jù)在生成是給定的過(guò)期時(shí)間,自動(dòng)地進(jìn)行定時(shí)撤銷。
WithDeadline是設(shè)置一個(gè)時(shí)間點(diǎn),時(shí)間對(duì)上了就到期。WithTimeout是設(shè)置一段時(shí)間,比如幾秒,過(guò)個(gè)這段時(shí)間,就超時(shí)。其實(shí)底層的WithTimeout也是通過(guò)WithDeadline實(shí)現(xiàn)的。WithTimeout的調(diào)用就等于WithDeadline(parent, time.Now().Add(timeout))(其中parent是父級(jí)context)
(3)WithValue
:入?yún)⑹歉讣?jí)parent,存儲(chǔ)的鍵和存儲(chǔ)的值。返回的是一個(gè)帶有數(shù)據(jù)的Context。這個(gè)Context是不能被撤銷的。撤銷的信號(hào)在傳遞的時(shí)候會(huì)跳過(guò)這個(gè)Context。
協(xié)程間共享數(shù)據(jù)
協(xié)程間共享數(shù)據(jù)主要使用的就是WithValue生成的子Context,這個(gè)Context存的值在其他的協(xié)程中也能讀取到。可以用做數(shù)據(jù)的共享。
超時(shí)取消協(xié)程
主要用到的是WithDeadline生成的子Context以及Go中HttpClient請(qǐng)求中的context字段(下文會(huì)有描述)
取消超時(shí)請(qǐng)求的模型
調(diào)度模型
其中,對(duì)于超時(shí)的判斷,是根據(jù)Context中的Done管道判斷的。如果超時(shí)了,則Done管道可以拿到東西。
超時(shí)之后取消請(qǐng)求
使用http.NewRequest方法獲取到的req,可以調(diào)用WithContext將定義好的WithTimeout類型的context放進(jìn)去,之后調(diào)用&htto.Client{}.Do()方法即可。網(wǎng)上有一些博客中讓手動(dòng)調(diào)用transport中的CancelRequest方法,但是這個(gè)方法已經(jīng)不被建議使用了。因?yàn)樗荒苋∠鸋ttp/2的請(qǐng)求。
現(xiàn)在在代碼中有一個(gè)私有化的roundTrip方法,會(huì)調(diào)用CancelRequest調(diào)用的cancelRequest方法。而這個(gè)roundTrip在transport中會(huì)在外面包一層RoundTrip方法,之后交給Client中的send方法進(jìn)行調(diào)用。(具體可以進(jìn)行源碼的查閱)。所以現(xiàn)在通過(guò)Client的Do方法,可以自動(dòng)完成請(qǐng)求的超時(shí)控制。
結(jié)論
該調(diào)度模型親測(cè)之后,確實(shí)可以實(shí)現(xiàn)請(qǐng)求的超時(shí)控制。只要在最外層設(shè)置超時(shí)時(shí)間時(shí)30s,只要過(guò)了30s,所有協(xié)程中的請(qǐng)求都會(huì)結(jié)束,對(duì)應(yīng)的協(xié)程也會(huì)相應(yīng)的結(jié)束;加上Client.Do方法,將超時(shí)控制變的更加簡(jiǎn)潔,后續(xù)會(huì)寫專門寫一篇關(guān)于http中Client的博客,詳細(xì)講解一下Client實(shí)現(xiàn)超時(shí)控制的原理。
到此這篇關(guān)于GoLang之使用Context控制請(qǐng)求超時(shí)的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)GoLang Context控制請(qǐng)求超時(shí) 內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go語(yǔ)言常見(jiàn)錯(cuò)誤之濫用getters/setters誤區(qū)實(shí)例探究
在Go語(yǔ)言編程中,恰如其分地使用getters和setters是至關(guān)重要的,過(guò)度和不適當(dāng)?shù)厥褂盟鼈兛赡軐?dǎo)致代碼冗余、可讀性差和封裝不當(dāng),在本文中,我們將深入探討如何識(shí)別濫用getter和setter的情況,以及如何采取最佳實(shí)踐來(lái)避免這些常見(jiàn)的Go錯(cuò)誤2024-01-01Go語(yǔ)言七篇入門教程七GC垃圾回收三色標(biāo)記
這篇文章主要為大家介紹了Go語(yǔ)言教程關(guān)于GC垃圾回收三色標(biāo)記的示例詳解,本篇文章是Go語(yǔ)言七篇入門教程系列文章,有需要的朋友可以借鑒參考下,希望能夠有所幫助2021-11-11Golang設(shè)計(jì)模式之生成器模式講解和代碼示例
生成器是一種創(chuàng)建型設(shè)計(jì)模式,使你能夠分步驟創(chuàng)建復(fù)雜對(duì)象,與其他創(chuàng)建型模式不同,生成器不要求產(chǎn)品擁有通用接口,這使得用相同的創(chuàng)建過(guò)程生成不同的產(chǎn)品成為可能,本文就通過(guò)代碼示例為大家詳細(xì)介紹Golang生成器模式,感興趣的同學(xué)可以參考下2023-06-06Go語(yǔ)言實(shí)現(xiàn)一個(gè)Http Server框架(二) Server的抽象
上一篇文章對(duì)http庫(kù)的基本使用做了說(shuō)明,這篇文章主要介紹了如何實(shí)現(xiàn)一個(gè)簡(jiǎn)單地httpServer,文中代碼示例非常詳細(xì),感興趣的朋友可以參考下2023-04-04Golang HTTP請(qǐng)求Json響應(yīng)解析方法以及解讀失敗的原因
這篇文章主要介紹了Golang HTTP請(qǐng)求Json響應(yīng)解析方法以及解讀失敗的原因,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-03-03