一文帶你搞懂Golang依賴注入的設(shè)計與實現(xiàn)
在現(xiàn)代的 web 框架里面,基本都有實現(xiàn)了依賴注入的功能,可以讓我們很方便地對應(yīng)用的依賴進(jìn)行管理,同時免去在各個地方 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ì)化的三種不同功能,分離出不同的接口可以讓我們的代碼更加的清晰,也會更利于代碼的后續(xù)演進(jìn)。
Injector:依賴注入容器Applicator:結(jié)構(gòu)體注入的接口Invoker:使用注入的依賴來調(diào)用函數(shù)TypeMapper:類型映射,需要特別注意的是,在Injector里面,是通過類型來綁定依賴(不同于Laravel的依賴注入容器可以通過字符串命名的方式來綁定依賴,當(dāng)然將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,然后在這里面定義一些在當(dāng)前協(xié)程以及當(dāng)前協(xié)程子協(xié)程可以用到的一些依賴,而不用影響外部的 Injector。
當(dāng)然上面說到的協(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 切片。
當(dāng)然,這只是其中一種使用依賴的方式,另外一種方式也比較常見,就是為結(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 才會進(jìn)行依賴注入
_, ok := structField.Tag.Lookup("inject")
// 字段支持反射設(shè)置,并且存在 inject tag 才會進(jìn)行注入
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)的詳細(xì)內(nèi)容,更多關(guān)于Golang依賴注入的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
淺析Go中函數(shù)的健壯性,panic異常處理和defer機制
這篇文章主要為大家詳細(xì)介紹了Go中函數(shù)的健壯性,panic異常處理和defer機制的相關(guān)知識,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2023-10-10
Go 循環(huán)結(jié)構(gòu)for循環(huán)使用教程全面講解
這篇文章主要為大家介紹了Go 循環(huán)結(jié)構(gòu)for循環(huán)使用全面講解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-10-10
Go json omitempty如何實現(xiàn)可選屬性
在Go語言中,使用`omitempty`可以幫助我們在進(jìn)行JSON序列化和反序列化時,忽略結(jié)構(gòu)體中的零值或空值,本文介紹了如何通過將字段類型改為指針類型,并在結(jié)構(gòu)體的JSON標(biāo)簽中添加`omitempty`來實現(xiàn)這一功能,例如,將float32修改為*float322024-09-09
go語言搬磚之go jmespath實現(xiàn)查詢json數(shù)據(jù)
這篇文章主要為大家介紹了go語言搬磚之go jmespath實現(xiàn)查詢json數(shù)據(jù),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06
使用Gin框架搭建一個Go Web應(yīng)用程序的方法詳解
在本文中,我們將要實現(xiàn)一個簡單的 Web 應(yīng)用程序,通過 Gin 框架來搭建,主要支持用戶注冊和登錄,用戶可以通過注冊賬戶的方式創(chuàng)建自己的賬號,并通過登錄功能進(jìn)行身份驗證,感興趣的同學(xué)跟著小編一起來看看吧2023-08-08

