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