GoFrame 如何優(yōu)雅的共享變量?Context的使用
前言
昨天merge代碼,發(fā)現(xiàn)了好多沖突,原因是同事在review項(xiàng)目,做鏈路追蹤,發(fā)現(xiàn)老項(xiàng)目有不少方法傳參不規(guī)范,沒有傳入Context,不方便做鏈路追蹤。
所以把這些方法和調(diào)用進(jìn)行了修改,導(dǎo)致了大量沖突,修復(fù)沖突又花了好長時(shí)間
所以:傳參規(guī)范還是要在項(xiàng)目啟動(dòng)時(shí)就確定好呀,一定要搞清楚Context怎么用呀!
今天就為大家介紹一下Context的使用:
告訴大家Context是什么?怎么用?為什么要用Context以及使用中的小技巧和注意問題。
Context是什么?
Context?指的是標(biāo)準(zhǔn)庫的context.Context?,是一個(gè)接口對(duì)象,常用于異步IO控制以及上下文流程變量的傳遞。
本文將要介紹的是Context如何優(yōu)雅的在業(yè)務(wù)流程中進(jìn)行變量的傳遞,以及為什么需要要進(jìn)行變量的傳遞。
為什么需要Context?
在Go?的執(zhí)行流程中,特別是HTTP/RPC?執(zhí)行流程中,沒有通過”全局變量”獲取請(qǐng)求參數(shù)的方式,只能通過上下文Context變量,傳遞到后續(xù)執(zhí)行流程的方法中。
如何使用?
Context上下文變量,包含了所有需要傳遞的共享變量。
并且Context中的共享變量是需要事先約定的,并且往往存儲(chǔ)為對(duì)象指針形式。
通過Context上下文,共享變量非常簡單,下面通過示例帶大家了解一下如何傳遞和使用通用的共享變量。
一、結(jié)構(gòu)定義
上下文對(duì)象中往往存儲(chǔ)一些需要共享的變量,這些變量通常使用結(jié)構(gòu)化的對(duì)象來存儲(chǔ),以方便維護(hù)。
例如,我們?cè)趍odel定義一個(gè)上下文中的共享變量:
介紹
- model.ContextKey?常量表示存儲(chǔ)在context.Context?上下文變量中的鍵名,該鍵名用于從傳遞的context.Context變量中存儲(chǔ)/獲取業(yè)務(wù)自定義的共享變量。
- model.Context?結(jié)構(gòu)體中的Session?表示當(dāng)前請(qǐng)求的Session?對(duì)象,在GoFrame?框架中每個(gè)HTTP?請(qǐng)求對(duì)象中都會(huì)有一個(gè)空的Session對(duì)象,該對(duì)象采用了懶初始化設(shè)計(jì),只有在真正執(zhí)行讀寫操作時(shí)才會(huì)初始化。
- model.Context?結(jié)構(gòu)體中的User?表示當(dāng)前登錄的用戶基本信息,只有在用戶登錄后才有數(shù)據(jù),否則是nil。
- model.Context?結(jié)構(gòu)體中的Data?,用于存儲(chǔ)自定義的KV?變量,因此一般來說開發(fā)者無需再往context.Context?上下文變量中增加自定義的鍵值對(duì),而是直接使用model.Context?對(duì)象的這個(gè)Data屬性即可。
二、邏輯封裝
由于該上下文對(duì)象也是和業(yè)務(wù)邏輯相關(guān)的,因此我們需要通過service對(duì)象將上下文變量封裝起來以方便其他模塊使用。
Tips
在架構(gòu)設(shè)計(jì)中,在哪個(gè)場景下設(shè)置Context是非常關(guān)鍵的。
上下文的變量必須在請(qǐng)求一開始便注入到請(qǐng)求流程中,以便于其他方法調(diào)用,所以在中間件中來實(shí)現(xiàn)是非常優(yōu)雅的選擇。
我們來看下面的介紹:
三、上下文變量注入
在HTTP?請(qǐng)求中我們可以使用GoFrame的中間件來實(shí)現(xiàn)。
在GRPC請(qǐng)求中我們也可以使用攔截器來實(shí)現(xiàn)。
在service?層的middleware管理對(duì)象中,我們可以這樣來定義:
這個(gè)中間件,初始化了用戶執(zhí)行流程 共享的對(duì)象,并且存儲(chǔ)到context.Context?變量中的對(duì)象是指針類型*model.Context。
這樣做的好處是:任何一個(gè)地方獲取到這個(gè)指針,不僅可以獲取到里面的數(shù)據(jù),而且能夠直接修改里面的數(shù)據(jù)。
TIPS
如果Session?中存在用戶登錄后的存儲(chǔ)信息,那么也會(huì)將需要共享的用戶基本信息寫入到*model.Context中。
四、上下文變量使用
方法定義
方法定義的第一個(gè)輸入?yún)?shù)往往預(yù)留給context.Context?類型參數(shù)使用,以便接受上下文變量,特別是service層的方法。
例如:
TIPS
另外一個(gè)好習(xí)慣是:方法的最后一個(gè)返回參數(shù)往往是error?類型。如果確定方法內(nèi)部永不會(huì)產(chǎn)生error,那么可以忽略。
?Context??對(duì)象獲取
通過service?中封裝的以下方法,將context.Context上下文變量傳遞進(jìn)去即可。
context.Context?上下文變量在GoFrame?框架的HTTP?請(qǐng)求中可以通過r.Context()方法獲取。
在GRPC?請(qǐng)求中,編譯生成的pb?文件中執(zhí)行方法的第一個(gè)參數(shù)即固定是context.Context。
自定義?Key-Value?
我們可以通過以下方式設(shè)置/獲取自定義的key-value鍵值對(duì)。
五、注意問題
上下文變量只傳遞必須的鏈路參數(shù)數(shù)據(jù),不要什么參數(shù)都往里面塞。特別是一些方法參數(shù)、傳參的數(shù)據(jù),千萬不能往上下文里面塞,而應(yīng)當(dāng)用顯示的方式傳遞方法參數(shù)。
上下文變量僅用作運(yùn)行時(shí)臨時(shí)使用,不可做持久化存儲(chǔ)長期使用。
總結(jié)
這篇文章詳細(xì)的為大家介紹了GoFrame上下文對(duì)象Context的知識(shí)點(diǎn):
Context的作用:在業(yè)務(wù)流程中進(jìn)行變量的共享。
Context的結(jié)構(gòu)定義、邏輯封裝、如何在中間件中注入、如何通過Context設(shè)置值和取值、Context如何自定義key-value、以及在項(xiàng)目開發(fā)中使用的注意問題。
歡迎Star GoFrame:https://github.com/gogf/gf
本文轉(zhuǎn)載自微信公眾號(hào)「 程序員升級(jí)打怪之旅」,作者「王中陽Go」,可以通過以下二維碼關(guān)注。
轉(zhuǎn)載本文請(qǐng)聯(lián)系「 程序員升級(jí)打怪之旅」公眾號(hào)。