1 什么是Context
最近在公司分析gRPC源碼,proto文件生成的代碼,接口函數(shù)第一個(gè)參數(shù)統(tǒng)一是ctx context.Context接口,公司不少同事都不了解這樣設(shè)計(jì)的出發(fā)點(diǎn)是什么,其實(shí)我也不了解其背后的原理。今天趁著妮妲臺(tái)風(fēng)妹子正面登陸深圳,全市停工、停課、停業(yè),在家休息找了一些資料研究把玩一把。
Context通常被譯作上下文,它是一個(gè)比較抽象的概念。在公司技術(shù)討論時(shí)也經(jīng)常會(huì)提到上下文。一般理解為程序單元的一個(gè)運(yùn)行狀態(tài)、現(xiàn)場、快照,而翻譯中上下又很好地詮釋了其本質(zhì),上下上下則是存在上下層的傳遞,上會(huì)把內(nèi)容傳遞給下。在Go語言中,程序單元也就指的是Goroutine。
每個(gè)Goroutine在執(zhí)行之前,都要先知道程序當(dāng)前的執(zhí)行狀態(tài),通常將這些執(zhí)行狀態(tài)封裝在一個(gè)Context變量中,傳遞給要執(zhí)行的Goroutine中。上下文則幾乎已經(jīng)成為傳遞與請求同生存周期變量的標(biāo)準(zhǔn)方法。在網(wǎng)絡(luò)編程下,當(dāng)接收到一個(gè)網(wǎng)絡(luò)請求Request,處理Request時(shí),我們可能需要開啟不同的Goroutine來獲取數(shù)據(jù)與邏輯處理,即一個(gè)請求Request,會(huì)在多個(gè)Goroutine中處理。而這些Goroutine可能需要共享Request的一些信息;同時(shí)當(dāng)Request被取消或者超時(shí)的時(shí)候,所有從這個(gè)Request創(chuàng)建的所有Goroutine也應(yīng)該被結(jié)束。
2 context包
Go的設(shè)計(jì)者早考慮多個(gè)Goroutine共享數(shù)據(jù),以及多Goroutine管理機(jī)制。Context介紹請參考Go Concurrency Patterns: Context,golang.org/x/net/context包就是這種機(jī)制的實(shí)現(xiàn)。
context包不僅實(shí)現(xiàn)了在程序單元之間共享狀態(tài)變量的方法,同時(shí)能通過簡單的方法,使我們在被調(diào)用程序單元的外部,通過設(shè)置ctx變量值,將過期或撤銷這些信號傳遞給被調(diào)用的程序單元。在網(wǎng)絡(luò)編程中,若存在A調(diào)用B的API, B再調(diào)用C的API,若A調(diào)用B取消,那也要取消B調(diào)用C,通過在A,B,C的API調(diào)用之間傳遞Context,以及判斷其狀態(tài),就能解決此問題,這是為什么gRPC的接口中帶上ctx context.Context參數(shù)的原因之一。
Go1.7(當(dāng)前是RC2版本)已將原來的golang.org/x/net/context包挪入了標(biāo)準(zhǔn)庫中,放在$GOROOT/src/context下面。標(biāo)準(zhǔn)庫中net、net/http、os/exec都用到了context。同時(shí)為了考慮兼容,在原golang.org/x/net/context包下存在兩個(gè)文件,go17.go是調(diào)用標(biāo)準(zhǔn)庫的context包,而pre_go17.go則是之前的默認(rèn)實(shí)現(xiàn),其介紹請參考go程序包源碼解讀。
context包的核心就是Context接口,其定義如下:
type Context interface { Deadline() (deadline time.Time, ok bool) Done() <-chan struct{} Err() error Value(key interface{}) interface{} }
Deadline會(huì)返回一個(gè)超時(shí)時(shí)間,Goroutine獲得了超時(shí)時(shí)間后,例如可以對某些io操作設(shè)定超時(shí)時(shí)間。
Done方法返回一個(gè)信道(channel),當(dāng)Context被撤銷或過期時(shí),該信道是關(guān)閉的,即它是一個(gè)表示Context是否已關(guān)閉的信號。
當(dāng)Done信道關(guān)閉后,Err方法表明Context被撤的原因。
Value可以讓Goroutine共享一些數(shù)據(jù),當(dāng)然獲得數(shù)據(jù)是協(xié)程安全的。但使用這些數(shù)據(jù)的時(shí)候要注意同步,比如返回了一個(gè)map,而這個(gè)map的讀寫則要加鎖。
Context接口沒有提供方法來設(shè)置其值和過期時(shí)間,也沒有提供方法直接將其自身撤銷。也就是說,Context不能改變和撤銷其自身。那么該怎么通過Context傳遞改變后的狀態(tài)呢?
3 context使用
無論是Goroutine,他們的創(chuàng)建和調(diào)用關(guān)系總是像層層調(diào)用進(jìn)行的,就像人的輩分一樣,而更靠頂部的Goroutine應(yīng)有辦法主動(dòng)關(guān)閉其下屬的Goroutine的執(zhí)行(不然程序可能就失控了)。為了實(shí)現(xiàn)這種關(guān)系,Context結(jié)構(gòu)也應(yīng)該像一棵樹,葉子節(jié)點(diǎn)須總是由根節(jié)點(diǎn)衍生出來的。
要?jiǎng)?chuàng)建Context樹,第一步就是要得到根節(jié)點(diǎn),context.Background函數(shù)的返回值就是根節(jié)點(diǎn):
func Background() Context
該函數(shù)返回空的Context,該Context一般由接收請求的第一個(gè)Goroutine創(chuàng)建,是與進(jìn)入請求對應(yīng)的Context根節(jié)點(diǎn),它不能被取消、沒有值、也沒有過期時(shí)間。它常常作為處理Request的頂層context存在。
有了根節(jié)點(diǎn),又該怎么創(chuàng)建其它的子節(jié)點(diǎn),孫節(jié)點(diǎn)呢?context包為我們提供了多個(gè)函數(shù)來創(chuàng)建他們:
func WithCancel(parent Context) (ctx Context, cancel CancelFunc) func WithDeadline(parent Context, deadline time.Time) (Context, CancelFunc) func WithTimeout(parent Context, timeout time.Duration) (Context, CancelFunc) func WithValue(parent Context, key interface{}, val interface{}) Context
函數(shù)都接收一個(gè)Context類型的參數(shù)parent,并返回一個(gè)Context類型的值,這樣就層層創(chuàng)建出不同的節(jié)點(diǎn)。子節(jié)點(diǎn)是從復(fù)制父節(jié)點(diǎn)得到的,并且根據(jù)接收參數(shù)設(shè)定子節(jié)點(diǎn)的一些狀態(tài)值,接著就可以將子節(jié)點(diǎn)傳遞給下層的Goroutine了。
再回到之前的問題:該怎么通過Context傳遞改變后的狀態(tài)呢?使用Context的Goroutine無法取消某個(gè)操作,其實(shí)這也是符合常理的,因?yàn)檫@些Goroutine是被某個(gè)父Goroutine創(chuàng)建的,而理應(yīng)只有父Goroutine可以取消操作。在父Goroutine中可以通過WithCancel方法獲得一個(gè)cancel方法,從而獲得cancel的權(quán)利。
第一個(gè)WithCancel函數(shù),它是將父節(jié)點(diǎn)復(fù)制到子節(jié)點(diǎn),并且還返回一個(gè)額外的CancelFunc函數(shù)類型變量,該函數(shù)類型的定義為:
type CancelFunc func()
調(diào)用CancelFunc對象將撤銷對應(yīng)的Context對象,這就是主動(dòng)撤銷Context的方法。在父節(jié)點(diǎn)的Context所對應(yīng)的環(huán)境中,通過WithCancel函數(shù)不僅可創(chuàng)建子節(jié)點(diǎn)的Context,同時(shí)也獲得了該節(jié)點(diǎn)Context的控制權(quán),一旦執(zhí)行該函數(shù),則該節(jié)點(diǎn)Context就結(jié)束了,則子節(jié)點(diǎn)需要類似如下代碼來判斷是否已結(jié)束,并退出該Goroutine:
select { case <-cxt.Done(): // do some clean... }
WithDeadline函數(shù)的作用也差不多,它返回的Context類型值同樣是parent的副本,但其過期時(shí)間由deadline和parent的過期時(shí)間共同決定。當(dāng)parent的過期時(shí)間早于傳入的deadline時(shí)間時(shí),返回的過期時(shí)間應(yīng)與parent相同。父節(jié)點(diǎn)過期時(shí),其所有的子孫節(jié)點(diǎn)必須同時(shí)關(guān)閉;反之,返回的父節(jié)點(diǎn)的過期時(shí)間則為deadline。
WithTimeout函數(shù)與WithDeadline類似,只不過它傳入的是從現(xiàn)在開始Context剩余的生命時(shí)長。他們都同樣也都返回了所創(chuàng)建的子Context的控制權(quán),一個(gè)CancelFunc類型的函數(shù)變量。
當(dāng)頂層的Request請求函數(shù)結(jié)束后,我們就可以cancel掉某個(gè)context,從而層層Goroutine根據(jù)判斷cxt.Done()來結(jié)束。
WithValue函數(shù),它返回parent的一個(gè)副本,調(diào)用該副本的Value(key)方法將得到val。這樣我們不光將根節(jié)點(diǎn)原有的值保留了,還在子孫節(jié)點(diǎn)中加入了新的值,注意若存在Key相同,則會(huì)被覆蓋。
3.1 小結(jié)
context包通過構(gòu)建樹型關(guān)系的Context,來達(dá)到上一層Goroutine能對傳遞給下一層Goroutine的控制。對于處理一個(gè)Request請求操作,需要采用context來層層控制Goroutine,以及傳遞一些變量來共享。
Context對象的生存周期一般僅為一個(gè)請求的處理周期。即針對一個(gè)請求創(chuàng)建一個(gè)Context變量(它為Context樹結(jié)構(gòu)的根);在請求處理結(jié)束后,撤銷此ctx變量,釋放資源。
每次創(chuàng)建一個(gè)Goroutine,要么將原有的Context傳遞給Goroutine,要么創(chuàng)建一個(gè)子Context并傳遞給Goroutine。
Context能靈活地存儲(chǔ)不同類型、不同數(shù)目的值,并且使多個(gè)Goroutine安全地讀寫其中的值。
當(dāng)通過父Context對象創(chuàng)建子Context對象時(shí),可同時(shí)獲得子Context的一個(gè)撤銷函數(shù),這樣父Context對象的創(chuàng)建環(huán)境就獲得了對子Context將要被傳遞到的Goroutine的撤銷權(quán)。
4 使用原則
Programs that use Contexts should follow these rules to keep interfaces consistent across packages and enable static analysis tools to check context propagation:
使用Context的程序包需要遵循如下的原則來滿足接口的一致性以及便于靜態(tài)分析。
在子Context被傳遞到的goroutine中,應(yīng)該對該子Context的Done信道(channel)進(jìn)行監(jiān)控,一旦該信道被關(guān)閉(即上層運(yùn)行環(huán)境撤銷了本goroutine的執(zhí)行),應(yīng)主動(dòng)終止對當(dāng)前請求信息的處理,釋放資源并返回。
Do not store Contexts inside a struct type; instead, pass a Context explicitly to each function that needs it. The Context should be the first parameter, typically named ctx;不要把Context存在一個(gè)結(jié)構(gòu)體當(dāng)中,顯式地傳入函數(shù)。Context變量需要作為第一個(gè)參數(shù)使用,一般命名為ctx;
Do not pass a nil Context, even if a function permits it. Pass context.TODO if you are unsure about which Context to use;即使方法允許,也不要傳入一個(gè)nil的Context,如果你不確定你要用什么Context的時(shí)候傳一個(gè)context.TODO;
Use context Values only for request-scoped data that transits processes and APIs, not for passing optional parameters to functions;使用context的Value相關(guān)方法只應(yīng)該用于在程序和接口中傳遞的和請求相關(guān)的元數(shù)據(jù),不要用它來傳遞一些可選的參數(shù);
The same Context may be passed to functions running in different goroutines; Contexts are safe for simultaneous use by multiple goroutines;同樣的Context可以用來傳遞到不同的goroutine中,Context在多個(gè)goroutine中是安全的;
編輯:黃飛
-
API
+關(guān)注
關(guān)注
2文章
1502瀏覽量
62080 -
函數(shù)
+關(guān)注
關(guān)注
3文章
4332瀏覽量
62666 -
網(wǎng)絡(luò)編程
+關(guān)注
關(guān)注
0文章
72瀏覽量
10075 -
go語言
+關(guān)注
關(guān)注
1文章
158瀏覽量
9050
原文標(biāo)題:理解GO CONTEXT機(jī)制
文章出處:【微信號:magedu-Linux,微信公眾號:馬哥Linux運(yùn)維】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
相關(guān)推薦
評論