0


goland单元测试

一、单元测试的概念

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.ProductLockInfo
    query.AddConditonsByOperator("ProductName", oquery.OP_EQUAL, productName)
    r = daoLockPtr.SearchByQuery(&allLockList, query)
    if !r.IsOk() {
        return
    }
    // 如果产品线没有查到锁记录,则改产品线锁状态为【未锁定】
    if len(allLockList) == 0 {
        status = LOCK_STATUS_UNLOCK
        return
    }
    // 遍历所有的锁信息
    for _, lockItem := range allLockList {
        // 一旦发现硬锁定,则立即返回产品线锁状态为【硬锁定】
        if lockItem.LockStatus == LOCK_STATUS_HARDLOCK {
            status = LOCK_STATUS_HARDLOCK
            return
        }
    }
    // 遍历完也不存在硬锁定,那就一定是【软锁定】
    status = LOCK_STATUS_SOFTLOCK
    return
}
快速生成单元测试代码

编写测试用例
参考1.2 中的第三点,我们需要涵盖到每一个分支,所以该单元测试至少要有下面三个测试用例
单元测试用例用例一:测试软锁定用例二:测试硬锁定用例三:测试未锁定输入SIODSIODSIOD输出软锁定 硬锁定未锁定
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       string
        args       args
        wantStatus string
        wantR      ocommon.ResultInfo
        mockData   []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的数据清除,不影响下一次mock
        patches.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

标签: 单元测试

本文转载自: https://blog.csdn.net/sunriseYJP/article/details/143914074
版权归原作者 大杯无糖 所有, 如有侵权,请联系我们删除。

“goland单元测试”的评论:

还没有评论