Go单元测试
简介
单元测试是go语言级别提供的完整功能,测试代码以*_test.go
命名,单元测试的case以Test
开头,性能测试case以Benchmark
开头,运行测试命令:go test <test_file_list>
例子
创建工程目录结构
目录结构说明参见规范-项目
└── src └── hmath ├── hmath.go └── hmath_test.go
实现排列组合函数
// src/hmath/hmath.go package hmath func combination(m,n int) int { if n > m-n { n = m - n } c := 1 for i := 0; i < n; i++ { c *= m - i c /= i + 1 } return c }
实现单元测试和性能测试
// src/hmath/hmath_test.go package hmath import ( "math/rand" "testing" ) // 单元测试 // 测试全局函数,以TestFunction命名 // 测试类成员函数,以TestClass_Function命名 func TestCombination(t *testing.T) { // 这里定义一个临时的结构体来存储测试case的参数以及期望的返回值 for _,unit := range []struct { m int n int expected int }{ {1,1},{4,1,4},2,6},3,4,{10,10},120},7,} { // 调用排列组合函数,与期望的结果比对,如果不一致输出错误 if actually := combination(unit.m,unit.n); actually != unit.expected { t.Errorf("combination: [%v],actually: [%v]",unit,actually) } } } // 性能测试 func BenchmarkCombination(b *testing.B) { // b.N会根据函数的运行时间取一个合适的值 for i := 0; i < b.N; i++ { combination(i+1,rand.Intn(i+1)) } } // 并发测试 func BenchmarkCombinationParallel(b *testing.B) { // 测试一个对象或者函数在多线程的场景下面是否安全 b.RunParallel(func(pb *testing.PB) { for pb.Next() { m := rand.Intn(100) + 1 n := rand.Intn(m) combination(m,n) } }) }
运行单元测试和性能测试
export GOPATH=$(pwd) go test src/hmath/*.go # 单元测试 go test --cover src/hmath/*.go # 单元测试覆盖率 go test -bench=. src/hmath/*.go # 性能测试
上面命令的输出如下:
hatlonely@localhost: ~/hatlonely/github/tmp $ go test src/hmath/hmath*.go ok command-line-arguments 0.005s hatlonely@localhost: ~/hatlonely/github/tmp $ go test --cover src/hmath/hmath*.go ok command-line-arguments 0.005s coverage: 100.0% of statements hatlonely@localhost: ~/hatlonely/github/tmp $ go test -bench=. src/hmath/*.go BenchmarkCombination-8 100000 217618 ns/op BenchmarkCombinationParallel-8 3000000 401 ns/op PASS ok command-line-arguments 23.599s