搞定Go單元測試(四)—— 依賴注入框架(wire)

在第一篇文章中提到過,爲了讓代碼可測,須要用依賴注入的方式來構建咱們的對象,而一般咱們會在main.go作依賴注入,這就致使main.go會愈來愈臃腫。爲了讓單元測試得以順利進行,main.go犧牲了它本應該纖細苗條的身材。太胖的main.go可不是什麼好的信號,本篇將介紹依賴注入框架(wire),致力於幫助main.go恢復身材。mysql

臃腫的main

main.go中作依賴注入,意味着在初始化代碼中咱們要管理:git

  1. 依賴的初始化順序
  2. 依賴之間的關係

對於小型項目而言,依賴的數量比較少,初始化代碼不會不少,不須要引入依賴注入框架。但對於依賴較多的中大型項目,初始化代碼又臭又長,可讀性和維護性變的不好,隨意感覺一下:程序員

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

What is wire?

wire是google開源的依賴注入框架。或者引用官方的話來講:「Wire is a code generation tool that automates connecting components using dependency injection」。sql

github.com/google/wirebash

Why wire?

除了wire,Go的依賴注入框架還有Uber的dig和Facebook的inject,它們都是使用反射機制來實現運行時依賴注入(runtime dependency injection),而wire則是採用代碼生成的方式來達到編譯時依賴注入(compile-time dependency injection)。使用反射帶來的性能損失卻是其次,更重要的是反射使得代碼難以追蹤和調試(反射會令Ctrl+左鍵失效...)。而wire生成的代碼是符合程序員常規使用習慣的代碼,十分容易理解和調試。
關於wire的優勢,在官方博文上有更詳細的的介紹:blog.golang.org/wire框架

How does it work?

本部份內容參考官方博文:blog.golang.org/wireide

wire有兩個基本的概念:provider和injector。函數

provider

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

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的呢?咱們須要寫一個函數來告訴它:

  • 定義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的步驟描述以下:

  1. 肯定所生成injector函數的函數簽名:func initUserStore(info ConnectionInfo) (*UserStore, error)
  2. 感知返回值第一個參數是*UserStore
  3. 檢查wire.Build列表,找到*UserStore的provider:NewUserStore
  4. 由函數簽名func NewUserStore(cfg *Config, db *mysql.DB)得知NewUserStore依賴於*Config, 和*mysql.DB
  5. 檢查wire.Build列表,找到*Config*mysql.DB的provider:NewDefaultConfigNewDB
  6. 由函數簽名func NewDefaultConfig() *Config得知*Config沒有其餘依賴了。
  7. 由函數簽名func NewDB(info *ConnectionInfo) (*mysql.DB, error)得知*mysql.DB依賴於ConnectionInfo
  8. 檢查wire.Build列表,找不到ConnectionInfo的provider,但在injector函數簽名中發現匹配的入參類型,直接使用該參數做爲NewDB的入參。
  9. 感知返回值第二個參數是error
  10. ....
  11. 按依賴關係,按順序調用provider函數,拼裝injector函數。

舉個栗子

栗子傳送門:wire-examples

注意

截止本文發佈前,官方代表wire的項目狀態是alpha,還不適合到生產環境,API存在變化的可能。
雖然是alpha,但其主要做用是爲咱們生成依賴注入代碼,其生成的代碼十分通俗易懂,在作好版本控制的前提下,即便是API發生變化,也不會對生成環境形成多壞的影響。我認爲仍是能夠放心使用的。

總結一下

本篇是本系列的最後一篇,回顧前幾篇文章,咱們以單元測試的原理與基本思想爲基礎,介紹了表格驅動測試方法,gomock,testify,wire這幾樣實用工具,經歷了「能寫單元測試」到「寫好單元測試」不斷優化的過程。但願本系列文章能讓你有所收穫。

相關文章
相關標籤/搜索