一文帶你搞懂Golang依賴注入的設(shè)計與實現(xiàn)
在現(xiàn)代的 web 框架里面,基本都有實現(xiàn)了依賴注入的功能,可以讓我們很方便地對應(yīng)用的依賴進行管理,同時免去在各個地方 new 對象的麻煩。比如 Laravel
里面的 Application
,又或者 Java 的 Spring
框架也自帶依賴注入功能。
今天我們來看看 go 里面實現(xiàn)依賴注入的一種方式,以 flamego 里的 inject 為例子。
我們要了解一個軟件的設(shè)計,先要看它定義了一個什么樣的模型,但是在了解模型之前,我們更應(yīng)該清楚了解,為什么會出現(xiàn)這個模型,也就是我們構(gòu)建出了這個模型到底是為了解決什么問題。
依賴注入要解決的問題
我們先來看看,在沒有依賴注入之前,我們需要的依賴是如何構(gòu)建出來的,假設(shè)有如下 struct
定義:
type A struct { } type B struct { a A } type C struct { b B } func test(c C) { println("c called") }
假設(shè)我們要調(diào)用 test
,就需要創(chuàng)建一個 C
的實例,而創(chuàng)建 C
的實例需要創(chuàng)建一個 B
的實例,而創(chuàng)建 B
的實例需要一個 A
的實例。如下是一個例子:
a := A{} b := B{a: a} c := C{b: b} test(c)
我們可以看到,這個過程非常的繁瑣,只有一個地方需要這樣調(diào)用 test
還好,如果有多個地方都需要調(diào)用 test
,那我們就要做很多創(chuàng)建實例的操作,而且一旦實例的構(gòu)建過程發(fā)生變化,我們就需要改動很多地方。
所以現(xiàn)在的 web 框架里面一般都將這個實例化的過程固化下來,在框架的某個地方注冊一些實例化的函數(shù),在我們需要的時候就調(diào)用之前注冊的實例化的函數(shù),實例化之后,再根據(jù)需要看看是否需要將這個實例保留在內(nèi)存里面,從而在免去了手動實例化的過程之外,節(jié)省我們資源的開銷(不用每次使用的時候都實例化一次)。
而這里說到的固化的實例化過程,其實就是我們本文所說的依賴注入。在 Laravel
里面我們可以通過 ServiceProvider
的 app()->register()
或者 app()->bind()
等函數(shù)來做依賴注入的一些操作。
inject 依賴注入模型/設(shè)計
以下是 Injector
的大概模型,Injector
接口里面嵌套了 Applicator
、Invoker
、TypeMapper
接口,之所以這樣做是出于接口隔離原則考慮,因為這三者代表了細化的三種不同功能,分離出不同的接口可以讓我們的代碼更加的清晰,也會更利于代碼的后續(xù)演進。
Injector
:依賴注入容器Applicator
:結(jié)構(gòu)體注入的接口Invoker
:使用注入的依賴來調(diào)用函數(shù)TypeMapper
:類型映射,需要特別注意的是,在Injector
里面,是通過類型來綁定依賴(不同于Laravel
的依賴注入容器可以通過字符串命名的方式來綁定依賴,當然將Injector
稍微改改也是可以實現(xiàn)的,就看有沒有這種需求罷了)。
// 依賴注入容器 type Injector interface { Applicator Invoker TypeMapper // 上一級 Injector SetParent(Injector) } // 給結(jié)構(gòu)體字段注入依賴 type Applicator interface { Apply(interface{}) error } // 調(diào)用函數(shù),Invoke 的參數(shù)是被調(diào)用的函數(shù), // 這個函數(shù)的參數(shù)事先通過 Injector 注入, // 調(diào)用的時候從 Injector 里面獲取依賴 type Invoker interface { Invoke(interface{}) ([]reflect.Value, error) } // 往 Injector 注入依賴 type TypeMapper interface { Map(...interface{}) TypeMapper MapTo(interface{}, interface{}) TypeMapper Set(reflect.Type, reflect.Value) TypeMapper Value(reflect.Type) reflect.Value }
表示成圖像大概如下:
我們可以通過 Injector
的 TypeMapper
來往依賴注入容器里面注入依賴,然后在我們需要為結(jié)構(gòu)體的字段注入依賴,又或者為函數(shù)參數(shù)注入依賴的時候,可以通過 Applicator
或者 Invoker
來實現(xiàn)注入依賴。
而 SetParent
這個方法比較有意思,它其實將 Injector
這個模型拓展了,形成了一個有父子關(guān)系的模型。在其他語言里面可能作用不是很明顯,但是在 go 里面,這個父子模型恰好和 go 的協(xié)程的父子模型一致。在 go 里面,我們可以在一個協(xié)程里面再創(chuàng)建一個 Injector
,然后在這里面定義一些在當前協(xié)程以及當前協(xié)程子協(xié)程可以用到的一些依賴,而不用影響外部的 Injector
。
當然上面說到的協(xié)程只是 Injector
里面 SetParent
的一種用法,另外一種用法是,我們的 web 應(yīng)用往往會根據(jù)路由前綴來劃分為不同的組,而這種路由組的結(jié)構(gòu)組織方式其實也是一種父子結(jié)構(gòu),在這種場景下,我們就可以針對全局注入一些依賴的情況下,再針對某個路由組來注入路由組特定的依賴。
injector 的依賴注入實現(xiàn)
我們來看看 injector
的結(jié)構(gòu)體:
type injector struct { // 注入的依賴 values map[reflect.Type]reflect.Value // 上級 Injector parent Injector }
這個結(jié)構(gòu)體定義很簡單,就只有兩個字段,values
和 parent
,我們通過 TypeMapper
注入的依賴都保存在 values
里面,values
是通過反射來記錄我們注入的參數(shù)類型和值的。
那我們是如何注入依賴的呢?再來看看 TypeMapper
的 Map
方法:
func (inj *injector) Map(values ...interface{}) TypeMapper { for _, val := range values { inj.values[reflect.TypeOf(val)] = reflect.ValueOf(val) } return inj }
我們可以看到,對于傳入給 Map
的參數(shù),這里獲取了它的反射類型作為 values
map 的 key,而獲取了傳入?yún)?shù)的反射值作為 values
里面 map 的值。其他的兩個方法 MapTo
、Set
也是類似的功能,最終的效果都是獲取依賴的類型作為 values 的 key,依賴的值作為 values 的值。
到此為止,我們知道 Injector
是如何注入依賴的了。
那么它又是如何去從依賴注入容器里面拿到我們注入的數(shù)據(jù)的呢?又是如何使用這些數(shù)據(jù)的呢?
我們再來看看 callInvoke
方法(也就是 Injector
的 Invoke
實現(xiàn)):
func (inj *injector) callInvoke(f interface{}, t reflect.Type, numIn int) ([]reflect.Value, error) { // 參數(shù)切片,用來保存從 Injector 里面獲取的依賴 var in []reflect.Value // 只有 f 有參數(shù)的時候,才需要從 Injector 獲取依賴 if numIn > 0 { // 初始化切片 in = make([]reflect.Value, numIn) var argType reflect.Type var val reflect.Value // 遍歷 f 參數(shù) for i := 0; i < numIn; i++ { // 獲取 f 參數(shù)類型 argType = t.In(i) // 從 Injector 獲取該類型對應(yīng)的依賴 val = inj.Value(argType) // 如果函數(shù)參數(shù)未注入,則調(diào)用出錯 if !val.IsValid() { return nil, fmt.Errorf("value not found for type %v", argType) } // 保存從 Injector 獲取到的值 in[i] = val } } // 通過反射調(diào)用 f 函數(shù),in 是參數(shù)切片 return reflect.ValueOf(f).Call(in), nil }
參數(shù)和返回值說明:
- 第一個參數(shù)是我們 Invoke 的函數(shù),這個函數(shù)的參數(shù),都會通過 Injector 根據(jù)函數(shù)參數(shù)類型獲取
- 第二個參數(shù) f 的反射類型,也就是 reflect.TypeOf(f)
- 第三個參數(shù)是 f 的參數(shù)個數(shù)
- 返回值是
reflect.Value
切片,如果我們在調(diào)用過程出錯,返回error
在這個函數(shù)中,會通過反射來獲取 f
的參數(shù)類型(reflect.Type
),拿到這個類型之后,從 Injector
里面獲取我們之前注入的依賴,這樣我們就可以拿到所有參數(shù)對應(yīng)的值。最后,通過 reflect.ValueOf(f)
來調(diào)用 f
函數(shù),參數(shù)是我們從 Injector
獲取到的值的切片。調(diào)用之后,返回函數(shù)調(diào)用結(jié)果,一個 reflect.Value
切片。
當然,這只是其中一種使用依賴的方式,另外一種方式也比較常見,就是為結(jié)構(gòu)體注入依賴,這跟 hyperf
里面通過注釋注解又或者 Spring
里面的注入方式有點類似。在 Injector
里面是通過 Apply
來為結(jié)構(gòu)體字段注入依賴的:
// 參數(shù) val 是待注入依賴的結(jié)構(gòu)體 func (inj *injector) Apply(val interface{}) error { v := reflect.ValueOf(val) // 獲取底層元素 for v.Kind() == reflect.Ptr { v = v.Elem() } // 底層類型不是結(jié)構(gòu)體則返回 if v.Kind() != reflect.Struct { return nil // Should not panic here ? } // v 的反射類型 t := v.Type() // 遍歷結(jié)構(gòu)體的字段 for i := 0; i < v.NumField(); i++ { // 獲取第 i 個結(jié)構(gòu)體字段 // v 的類型是 reflect.Value // v.Field 返回的是結(jié)構(gòu)體字段的值 f := v.Field(i) // t 的類型是 *reflect.rtype // t.Field 返回的是 reflect.Type,是類型信息 structField := t.Field(i) // 檢查是否有 inject tag,有這個 tag 才會進行依賴注入 _, ok := structField.Tag.Lookup("inject") // 字段支持反射設(shè)置,并且存在 inject tag 才會進行注入 if f.CanSet() && ok { // 通過反射類型從 Injector 中獲取對應(yīng)的值 ft := f.Type() v := inj.Value(ft) // 獲取不到注入的依賴,則返回錯誤 if !v.IsValid() { return fmt.Errorf("value not found for type %v", ft) } // 設(shè)置結(jié)構(gòu)體字段值 f.Set(v) } } return nil }
簡單來說,Injector
里面,通過 TypeMapper
來注入依賴,然后通過 Apply
或者 Invoke
來使用注入的依賴。
例子
還是以一開始的例子為例,通過依賴注入的方式來改造一下:
a := A{} b := B{a: a} c := C{b: b} // 新建依賴注入容器 inj := injector{ values: make(map[reflect.Type]reflect.Value), } // 注入依賴 c inj.Map(c) // 調(diào)用函數(shù) test,test 的參數(shù) `C` 會通過依賴注入容器獲取 _, _ = inj.Invoke(test) // 輸出 "c called"
這個例子中,我們通過 inj.Map
來注入了依賴,在后續(xù)通過 inj.Invoke
來調(diào)用 test
函數(shù)的時候,將會從依賴注入容器里面獲取 test
的參數(shù),然后將這些參數(shù)傳入 test
來調(diào)用。
這個例子也許比較簡單,但是如果我們很多地方都需要用到 C
這個參數(shù)的話,我們通過 inj.Invoke
的方式來調(diào)用函數(shù)就可以避免每一次調(diào)用都要實例化 C
的繁瑣操作了。
以上就是一文帶你搞懂Golang依賴注入的設(shè)計與實現(xiàn)的詳細內(nèi)容,更多關(guān)于Golang依賴注入的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
淺析Go中函數(shù)的健壯性,panic異常處理和defer機制
這篇文章主要為大家詳細介紹了Go中函數(shù)的健壯性,panic異常處理和defer機制的相關(guān)知識,文中的示例代碼講解詳細,感興趣的小伙伴可以了解一下2023-10-10Go 循環(huán)結(jié)構(gòu)for循環(huán)使用教程全面講解
這篇文章主要為大家介紹了Go 循環(huán)結(jié)構(gòu)for循環(huán)使用全面講解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-10-10Go json omitempty如何實現(xiàn)可選屬性
在Go語言中,使用`omitempty`可以幫助我們在進行JSON序列化和反序列化時,忽略結(jié)構(gòu)體中的零值或空值,本文介紹了如何通過將字段類型改為指針類型,并在結(jié)構(gòu)體的JSON標簽中添加`omitempty`來實現(xiàn)這一功能,例如,將float32修改為*float322024-09-09go語言搬磚之go jmespath實現(xiàn)查詢json數(shù)據(jù)
這篇文章主要為大家介紹了go語言搬磚之go jmespath實現(xiàn)查詢json數(shù)據(jù),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-06-06使用Gin框架搭建一個Go Web應(yīng)用程序的方法詳解
在本文中,我們將要實現(xiàn)一個簡單的 Web 應(yīng)用程序,通過 Gin 框架來搭建,主要支持用戶注冊和登錄,用戶可以通過注冊賬戶的方式創(chuàng)建自己的賬號,并通過登錄功能進行身份驗證,感興趣的同學跟著小編一起來看看吧2023-08-08