一、单元测试的概念
1.1 什么是单元测试,有什么用?
单元测试是针对于函数的测试,用来保证该函数的逻辑正确性。
1.2 单元测试的要求?
1. 单元测试在正式上线之前应该全部自动执行,并且需要保证全部通过
2. 单元测试需要构建 【输入】 和 【预期输出】的case,case需要靠人工构建,涵盖各种边界情况
3. 单元测试需要测试到代码的每一个逻辑分支
4. 单元测试关注的是代码逻辑是否正确,无需关注网络调用、数据查询等,对于此部分代码可以mock掉
5. 单元测试行覆盖率要达到70%,函数覆盖率要达到100%(controller层除外)
1.3 单元测试的常见误区
1. 单元测试行覆盖率100%只能保证写的测试用例走过了所有的代码,但不能保证代码逻辑完全无误
1.4 gomock工具-gomonkey
官方文档 :https://github.com/agiledragon/gomonkey
使用教程:https://cloud.tencent.com/developer/article/1872029
https://www.ddhigh.com/2021/09/18/gomonkey-private-method-stub/ 解决了无法打桩私有方法的问题
二、单元测试实战
func GetFinalStatus(productName string) (status string, r ocommon.ResultInfo) {daoLockPtr := dao.CreateProductLockInfoPtr()query := oquery.NewQueryStructOfTable()var allLockList []dao.ProductLockInfoquery.AddConditonsByOperator("ProductName", oquery.OP_EQUAL, productName)r = daoLockPtr.SearchByQuery(&allLockList, query)if !r.IsOk() {return}// 如果产品线没有查到锁记录,则改产品线锁状态为【未锁定】if len(allLockList) == 0 {status = LOCK_STATUS_UNLOCKreturn}// 遍历所有的锁信息for _, lockItem := range allLockList {// 一旦发现硬锁定,则立即返回产品线锁状态为【硬锁定】if lockItem.LockStatus == LOCK_STATUS_HARDLOCK {status = LOCK_STATUS_HARDLOCKreturn}}// 遍历完也不存在硬锁定,那就一定是【软锁定】status = LOCK_STATUS_SOFTLOCKreturn
}
快速生成单元测试代码
编写测试用例
参考1.2 中的第三点,我们需要涵盖到每一个分支,所以该单元测试至少要有下面三个测试用例
单元测试用例 | 用例一:测试软锁定 | 用例二:测试硬锁定 | 用例三:测试未锁定 |
输入 | SIOD | SIOD | SIOD |
输出 | 软锁定 | 硬锁定 | 未锁定 |
mock数据并完成测试代码
参数1.2 中的第四点,该方法的单元测试关注的是代码逻辑是否正确,对于数据的来源不关注,再加上1.2 中的第一点,单元测试需要在每次提交代码之前,全部执行通过
所以我们需要mock掉 daoLockPtr.SearchByQuery 方法,并且将allLockList 变量用mock数据替代
如果不mock掉daoLockPtr.SearchByQuery,而是每次都去从数据库中查询,不可能同时通过上述三个测试用例
此时就用到了上面提到的测试工具,对daoLockPtr.SearchByQuery(&allLockList, query) 方法进行打桩 并mock掉 allLockList数据
具体实现如下方代码
func TestGetFinalStatus(t *testing.T) {type args struct {productName string}tests := []struct {name stringargs argswantStatus stringwantR ocommon.ResultInfomockData []dao.LockInfo // 此变量存储每一个测试用例的mock数据}{// TODO: Add test cases.{name: "未锁定",args: args{productName: "SIOD"},wantStatus: "unLock",wantR: ocommon.ResultInfo{ErrNo: 0},mockData: nil, // 对于未锁定,mockData就是nil},{name: "软锁定",args: args{productName: "SIOD"},wantStatus: "softLock",wantR: ocommon.ResultInfo{ErrNo: 0},mockData: []dao.LockInfo{dao.LockInfo{LockStatus: LOCK_STATUS_SOFTLOCK}},// 对于软锁定,mockData的一种情况就是一条锁的状态为 【软锁】},{name: "硬锁定",args: args{productName: "SIOD"},wantStatus: "hardLock",wantR: ocommon.ResultInfo{ErrNo: 0},mockData: []dao.LockInfo{dao.LockInfo{LockStatus: LOCK_STATUS_HARDLOCK}},// 对于硬锁定,mockData的一种情况就是一条锁的状态为 【硬锁】},}for _, tt := range tests {// 使用 gomonkey 来 mock 方法patches := gomonkey.ApplyMethod(reflect.TypeOf(&dbBase.DbBase{}), "SearchByQuery", func(_ *dbBase.DbBase, allLockList interface{}, query *oquery.QueryStructOfTable) ocommon.ResultInfo {if data, ok := allLockList.(*[]dao.LockInfo); ok {*data = tt.mockData}return ocommon.ResultInfo{}})t.Run(tt.name, func(t *testing.T) {gotStatus, gotR := GetFinalStatus(tt.args.productName)if gotStatus != tt.wantStatus {t.Errorf("GetFinalStatus() gotStatus = %v, want %v", gotStatus, tt.wantStatus)}if !reflect.DeepEqual(gotR.ErrNo, tt.wantR.ErrNo) {t.Errorf("GetFinalStatus() gotR = %v, want %v", gotR.ErrNo, tt.wantR.ErrNo)}})// 每一个测试用例结束后,将mock的数据清除,不影响下一次mockpatches.Reset()}
}
查看单元测试覆盖率
我们在该方法所在的目录下面执行下面的命令,然后将单元测试的信息输出到目录下的c.out
执行单元测试
GOOS=darwin GOARCH=amd64 go test -cover -coverprofile=c.out -gcflags="all=-N -l" -run 'TestGetFinalStatus'
查看单元测试覆盖率,执行完之后,会自动打开一个html网页
go tool cover -html=c.out
从下图中,可以看出,我们在lock_info.go文件的单元测试覆盖率只有10.3%,我们需要把该文件中的其他方法根据项目实际情况补充单元测试
该方法中只有一行代码没有覆盖到,就是数据库查询错误,单元测试的目的是确保后续的代码逻辑正确,所以这一行代码可以不做测试
如何写出更加完备的单元测试?
上面代码可以看到对于该方法的覆盖已经达到每一个分支(数据库错误处理分支除外),但是其并不能保证其代码逻辑一定没有问题。
所以我们需要再补充几个单元测试用例,正如1.2 中的第二条一样,单元测试的难点就是需要人工去构思各种边界情况
{name: "混合锁取最高级别",args: args{productName: "SIOD"},wantStatus: "hardLock",wantR: ocommon.ResultInfo{ErrNo: 0},mockData: []dao.ProductLockInfo{{LockStatus: LOCK_STATUS_HARDLOCK}, {LockStatus: LOCK_STATUS_SOFTLOCK}}, // 此处的mockData 是一个硬锁,一个软锁
},
{name: "多把相同状态的锁",args: args{productName: "SIOD"},wantStatus: "hardLock",wantR: ocommon.ResultInfo{ErrNo: 0},mockData: []dao.ProductLockInfo{{LockStatus: LOCK_STATUS_HARDLOCK}, {LockStatus: LOCK_STATUS_HARDLOCK}}, // 此处的mockData 是一个硬锁,一个软锁
},
参考文章
https://www.liwenzhou.com/posts/Go/unit-test-0/#c-0-1-8
https://time.geekbang.org/column/article/10275?utm_campaign=geektime_search&utm_content=geektime_search&utm_medium=geektime_search&utm_source=geektime_search&utm_term=geektime_search