在第一篇文章中提到過,爲了讓代碼可測,須要用依賴注入的方式來構建咱們的對象,而一般咱們會在main.go
作依賴注入,這就致使main.go
會愈來愈臃腫。爲了讓單元測試得以順利進行,main.go
犧牲了它本應該纖細苗條的身材。太胖的main.go
可不是什麼好的信號,本篇將介紹依賴注入框架(wire),致力於幫助main.go
恢復身材。mysql
在main.go
中作依賴注入,意味着在初始化代碼中咱們要管理:git
對於小型項目而言,依賴的數量比較少,初始化代碼不會不少,不須要引入依賴注入框架。但對於依賴較多的中大型項目,初始化代碼又臭又長,可讀性和維護性變的不好,隨意感覺一下:程序員
func main() {
config := NewConfig()
// db依賴配置
db, err := ConnectDatabase(config)
if err != nil {
panic(err)
}
// PersonRepository 依賴db
personRepository := NewPersonRepository(db)
// PersonService 依賴配置 和 PersonRepository
personService := NewPersonService(config, personRepository)
// NewServer 依賴配置和PersonService
server := NewServer(config, personService)
server.Run()
}
複製代碼
實踐代表,修改有大量依賴關係的初始化代碼是一項乏味且耗時的工做。這個時候,咱們就須要依賴注入框架來幫忙,簡化初始化代碼。github
上述代碼來自:blog.drewolson.org/dependency-…golang
wire是google開源的依賴注入框架。或者引用官方的話來講:「Wire is a code generation tool that automates connecting components using dependency injection」。sql
除了wire,Go的依賴注入框架還有Uber的dig和Facebook的inject,它們都是使用反射機制來實現運行時依賴注入(runtime dependency injection),而wire則是採用代碼生成的方式來達到編譯時依賴注入(compile-time dependency injection)。使用反射帶來的性能損失卻是其次,更重要的是反射使得代碼難以追蹤和調試(反射會令Ctrl+左鍵失效...)。而wire生成的代碼是符合程序員常規使用習慣的代碼,十分容易理解和調試。
關於wire的優勢,在官方博文上有更詳細的的介紹:blog.golang.org/wire框架
本部份內容參考官方博文:blog.golang.org/wireide
wire有兩個基本的概念:provider和injector。函數
provider就是普通的Go函數,能夠把它看做是某對象的構造函數,咱們經過provider告訴wire該對象的依賴狀況:
// NewUserStore是*UserStore的provider,代表*UserStore依賴於*Config和 *mysql.DB.
func NewUserStore(cfg *Config, db *mysql.DB) (*UserStore, error) {...}
// NewDefaultConfig是*Config的provider,沒有依賴
func NewDefaultConfig() *Config {...}
// NewDB是*mysql.DB的provider,依賴於ConnectionInfo
func NewDB(info ConnectionInfo) (*mysql.DB, error) {...}
// UserStoreSet 可選項,可使用wire.NewSet將一般會一塊兒使用的依賴組合起來。
var UserStoreSet = wire.NewSet(NewUserStore, NewDefaultConfig)
複製代碼
injector是wire生成的函數,咱們經過調用injector來獲取咱們所需的對象或值,injector會按照依賴關係,按順序調用provider函數:
// File: wire_gen.go
// Code generated by Wire. DO NOT EDIT.
//go:generate wire
//+build !wireinject
// initUserStore是由wire生成的injector
func initUserStore(info ConnectionInfo) (*UserStore, error) {
// *Config的provider函數
defaultConfig := NewDefaultConfig()
// *mysql.DB的provider函數
db, err := NewDB(info)
if err != nil {
return nil, err
}
// *UserStore的provider函數
userStore, err := NewUserStore(defaultConfig, db)
if err != nil {
return nil, err
}
return userStore, nil
}
複製代碼
injector幫咱們把按順序初始化依賴的步驟給作了,咱們在main.go
中只須要調用initUserStore
方法就能獲得咱們想要的對象了。
那麼wire是怎麼知道如何生成injector的呢?咱們須要寫一個函數來告訴它:
wire.Build
方法列舉生成injector所需的provider例如:
// initUserStore用於聲明injector的函數簽名
func initUserStore(info ConnectionInfo) (*UserStore, error) {
// wire.Build聲明要獲取一個UserStore須要調用到哪些provider函數
wire.Build(UserStoreSet, NewDB)
return nil, nil // 這些返回值wire並不關心。
}
複製代碼
有了上面的函數,wire就能夠得知如何生成injector了。wire生成injector的步驟描述以下:
func initUserStore(info ConnectionInfo) (*UserStore, error)
*UserStore
wire.Build
列表,找到*UserStore
的provider:NewUserStore
func NewUserStore(cfg *Config, db *mysql.DB)
得知NewUserStore
依賴於*Config
, 和*mysql.DB
wire.Build
列表,找到*Config
和*mysql.DB
的provider:NewDefaultConfig
和NewDB
func NewDefaultConfig() *Config
得知*Config
沒有其餘依賴了。func NewDB(info *ConnectionInfo) (*mysql.DB, error)
得知*mysql.DB
依賴於ConnectionInfo
。wire.Build
列表,找不到ConnectionInfo
的provider,但在injector函數簽名中發現匹配的入參類型,直接使用該參數做爲NewDB
的入參。error
栗子傳送門:wire-examples
截止本文發佈前,官方代表wire的項目狀態是alpha,還不適合到生產環境,API存在變化的可能。
雖然是alpha,但其主要做用是爲咱們生成依賴注入代碼,其生成的代碼十分通俗易懂,在作好版本控制的前提下,即便是API發生變化,也不會對生成環境形成多壞的影響。我認爲仍是能夠放心使用的。
本篇是本系列的最後一篇,回顧前幾篇文章,咱們以單元測試的原理與基本思想爲基礎,介紹了表格驅動測試方法,gomock,testify,wire這幾樣實用工具,經歷了「能寫單元測試」到「寫好單元測試」不斷優化的過程。但願本系列文章能讓你有所收穫。