摘要:在剛開始的時候,我通常都會先啟動一個簡單的服務(wù),然后再運行我的單元測試??墒沁@個單元測試測起來似乎非常笨重。王尼瑪張全蛋測試之這是自定義的返回值,可以是任意類型。
在開發(fā)過程中往往需要配合單元測試,但是很多時候,單元測試需要依賴一些比較復雜的準備工作,比如需要依賴數(shù)據(jù)庫環(huán)境,需要依賴網(wǎng)絡(luò)環(huán)境,單元測試就變成了一件非常麻煩的事情。舉例來說,比如我們需要請求一個網(wǎng)頁,并將請求回來的數(shù)據(jù)進行處理。在剛開始的時候,我通常都會先啟動一個簡單的http服務(wù),然后再運行我的單元測試??墒沁@個單元測試測起來似乎非常笨重。甚至在持續(xù)集成過程中,我還為了能夠自動化測試,特意寫了一個腳本自動啟動相應的服務(wù)。事情似乎需要進行一些改變。
mock對象就是為了解決上面的問題而誕生的,mock(模擬)對象能夠模擬實際依賴對象的功能,同時又不需要非常復雜的準備工作,你需要做的,僅僅就是定義對象接口,然后實現(xiàn)它,再交給測試對象去使用。
go-mock是專門為go語言開發(fā)的mock庫,該庫使用方式簡單,支持自動生成代碼,可以說是不可多得的好工具。下面我就簡單地展示一下go-mock是如何工作的:
首先你需要做的是將依賴下載到本地:
go get github.com/golang/mock/gomock go get github.com/golang/mock/mockgen
第一個是代碼依賴,第二個是命令行工具(特別好用)。
下面用一個非常簡單的例子來說明gomock是如何工作的:
我在$GOPATH/src目錄下新建一個項目:hellomock,在$GOPATH/src/hellomock目錄下新建hellomock.go,并定義一個接口Talker:
package hellomock type Talker interface { SayHello(word string)(response string) }
然后我們需要一個實現(xiàn)了Talker功能的結(jié)構(gòu)體,假設(shè)我們有這樣的場景,我們現(xiàn)在有一個迎賓的崗位,需要一個能夠迎賓的迎賓員,當然這個迎賓員可以是一個人,或者是一只鸚鵡。那么我們需要做的是,定義一個Persion結(jié)構(gòu)體(或者是鸚鵡或者是別的什么東西),并實現(xiàn)Talker接口:
person.go
package hellomock import "fmt" type Person struct{ name string } func NewPerson(name string)*Person{ return &Person{ name:name, } } func (p *Person)SayHello(name string)(word string) { return fmt.Sprintf("Hello %s, welcome come to our company, my name is %s",name,p.name) }
現(xiàn)在我們的Person已經(jīng)實現(xiàn)了Talker接口,現(xiàn)在我們讓他發(fā)揮作用吧!
現(xiàn)在假設(shè),我們有一個公司,公司有一個迎賓員,也就是我們的前臺妹子,這個妹子實現(xiàn)了Talker接口.她能夠自動向來的客人SayHello:
company.go
package hellomock type Company struct { Usher Talker } func NewCompany(t Talker) *Company{ return &Company{ Usher:t, } } func ( c *Company) Meeting(gusetName string)string{ return c.Usher.SayHello(gusetName) }
我們的場景已經(jīng)設(shè)計好了,那么我們傳統(tǒng)的話,會如何實現(xiàn)單元測試呢?
company_test.go
package hellomock import "testing" func TestCompany_Meeting(t *testing.T) { person := NewPerson("王尼美") company := NewCompany(person) t.Log(company.Meeting("王尼瑪")) }
測試之:
/usr/local/go/bin/go test -v hellomock -run ^TestCompany_Meeting$ company_test.go:8: Hello 王尼瑪, welcome come to our company, my name is 王尼美 ok hellomock 0.013s
現(xiàn)在我們構(gòu)造一個王尼美還是很簡單的,但是我們現(xiàn)在要用mock對象進行模擬,這時mockgen就登場了:
? hellomock> mkdir mock ? hellomock> mockgen -source=hellomock.go > mock/mock_Talker.go
這個時候,將會生成mock/mock_Talker.go文件:
需要注意的是,自動生成的文件同源文件在不同的包下,需要新建一個目錄存放
我們并不需要在意生成文件的內(nèi)容,我們只需要知道如何去使用即可
mock_Talker.go
// Automatically generated by MockGen. DO NOT EDIT! // Source: hellomock.go package mock_hellomock import ( gomock "github.com/golang/mock/gomock" ) // MockTalker is a mock of Talker interface type MockTalker struct { ctrl *gomock.Controller recorder *MockTalkerMockRecorder } // MockTalkerMockRecorder is the mock recorder for MockTalker type MockTalkerMockRecorder struct { mock *MockTalker } // NewMockTalker creates a new mock instance func NewMockTalker(ctrl *gomock.Controller) *MockTalker { mock := &MockTalker{ctrl: ctrl} mock.recorder = &MockTalkerMockRecorder{mock} return mock } // EXPECT returns an object that allows the caller to indicate expected use func (_m *MockTalker) EXPECT() *MockTalkerMockRecorder { return _m.recorder } // SayHello mocks base method func (_m *MockTalker) SayHello(name string) string { ret := _m.ctrl.Call(_m, "SayHello", name) ret0, _ := ret[0].(string) return ret0 } // SayHello indicates an expected call of SayHello func (_mr *MockTalkerMockRecorder) SayHello(arg0 interface{}) *gomock.Call { return _mr.mock.ctrl.RecordCall(_mr.mock, "SayHello", arg0) }
接下來看看如何去使用這個mock對象,新建一個單元測試:
func TestCompany_Meeting2(t *testing.T) { ctl := gomock.NewController(t) mock_talker := mock_hellomock.NewMockTalker(ctl) mock_talker.EXPECT().SayHello(gomock.Eq("王尼瑪")).Return("這是自定義的返回值,可以是任意類型。") company := NewCompany(mock_talker) t.Log(company.Meeting("王尼瑪")) //t.Log(company.Meeting("張全蛋")) }
測試之:
/usr/local/go/bin/go test -v hellomock -run ^TestCompany_Meeting2$ company_test.go:21: 這是自定義的返回值,可以是任意類型。 ok hellomock 0.015s
可以看到,返回的是我們在mock對象上定義的返回值。
需要說明的一點是,mock對象的SayHello可以接受的參數(shù)有gomock.Eq(x interface{})和gomock.Any(),前一個要求測試用例入餐嚴格符合x,第二個允許傳入任意參數(shù)。比如我們在注釋掉的測試中傳入了"張全蛋",結(jié)果報錯,測試失敗:
/usr/local/go/bin/go test -v hellomock -run ^TestCompany_Meeting2$ controller.go:113: no matching expected call: *mock_hellomock.MockTalker.SayHello([張全蛋]) exit status 1 FAIL hellomock 0.007s
本文作為拋磚引玉,gomock還有很多高級用法,希望大家能夠自行探索。
參考文章:
https://github.com/golang/moc...
https://github.com/grpc/grpc-...
轉(zhuǎn)載請注明出處 http://www.chenquan.me
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/8777.html
摘要:使用進行單元測試原文地址使用進行單元測試前言在實際項目中,需要進行單元測試的時候。卻往往發(fā)現(xiàn)有一大堆依賴項。這時候就是大顯身手的時候了是語言的一個框架,官方的那種 使用 Gomock 進行單元測試 原文地址:使用 Gomock 進行單元測試 前言 在實際項目中,需要進行單元測試的時候。卻往往發(fā)現(xiàn)有一大堆依賴項。這時候就是 Gomock 大顯身手的時候了 Gomock 是 Go 語言的一...
摘要:接下來我們將介紹如何對對象的方法進行模擬測試。選項創(chuàng)建模擬測試接口我們可以在的構(gòu)造函數(shù)中提供一個模擬測試實例,而不是模擬創(chuàng)建具體的模擬測試方法。 如何不靠耐心測試 通常,我們編寫的軟件會直接與那些我們稱之為骯臟的服務(wù)交互。通俗地說,服務(wù)對我們的應用來說是至關(guān)重要的,它們之間的交互是我們設(shè)計好的,但這會帶來我們不希望的副作用——就是那些在我們自己測試的時候不希望的功能。 比如,可能我們...
摘要:即便是提供測試環(huán)境的外部系統(tǒng),一般也僅在開發(fā)聯(lián)調(diào)階段配合提供聯(lián)調(diào)測試對接服務(wù),一旦聯(lián)調(diào)測試結(jié)束,也不再繼續(xù)提供測試服務(wù)。 MOCK API 的定義 根據(jù)百度百科的定義,mock測試就是在測試過程中,對于某些不容易構(gòu)造或者不容易獲取的對象,用一個虛擬的對象來創(chuàng)建以便測試的測試方法。這個虛擬的對象就是mock對象,mock對象就是真實對象在調(diào)試期間的代替品。 在瀑布流開發(fā)模式中,如果前端開...
摘要:本文適用的場景在對移動端的純移動端功能或者前端頁面的純前端功能進行測試時,服務(wù)端接口返回的數(shù)據(jù)不滿足要求,或者制造測試數(shù)據(jù)比較復雜,需要使用方法來快速構(gòu)造數(shù)據(jù)。進入官網(wǎng)后,首先創(chuàng)建一個項目,一個項目包含若干個接口,我們最終模擬的是接口。 本文適用的場景:在對移動端APP的純移動端功能或者前端H5頁面的純前端功能進行測試時,服務(wù)端接口返回的數(shù)據(jù)不滿足要求,或者制造測試數(shù)據(jù)比較復雜,需要使...
摘要:準確的說,是中一個用于支持單元測試的庫,它的主要功能是使用對象替代掉指定的對象,以達到模擬對象的行為。下面我們使用對象在單元測試中分別測試訪問正常和訪問不正常的情況。 Mock是什么 Mock這個詞在英語中有模擬的這個意思,因此我們可以猜測出這個庫的主要功能是模擬一些東西。準確的說,Mock是Python中一個用于支持單元測試的庫,它的主要功能是使用mock對象替代掉指定的Python...
閱讀 2465·2021-11-22 09:34
閱讀 3072·2021-10-25 09:43
閱讀 1987·2021-10-11 10:59
閱讀 3396·2021-09-22 15:13
閱讀 2334·2021-09-04 16:40
閱讀 426·2019-08-30 15:53
閱讀 3196·2019-08-30 11:13
閱讀 2610·2019-08-29 17:30