Go语言单元测试
1. 概述
单元测试(unit testing)
是指对软件 中的 最小可测试单元进行检查和验证Go语言自身提供了一套轻量级的单元测试和性能测试系统,符合规则的测试代码会在运行测试时被自动识别执行
2. 基础测试
看看一个基本的测试案例是怎么写的
文件结构如下
| |_main.go |_utils |___func1.go |___func1_test.go |___sham_test.go
func1.go
package utils
func AddTwoNums(x, y int) (sum int) {
sum = x + y
return
}
func AddUp(x int) int {
if x < 2 {
return x
}
sum := 0
for i := 0; i <= x; i++ {
sum += i
}
return sum
}
main.go
package main
import (
"utils"
"fmt"
)
func main(){
// 调用函数AddtwoNums
sum := utils.AddTwoNums(9,8)
fmt.Println(sum) // 17
// 调用函数AddUp
sum1 := utils.AddUp(100)
fmt.Println(sum1) // 5050
}
接下来我们看看测试用例怎么写
func1_test.go
package utils
import (
"testing" // 导入testing包
)
// AddTwoNums的测试用例1
// 测试AddTwoNums函数 结果正确
func TestAddTwoNums1(t *testing.T) {
sum := AddTwoNums(9,8)
if sum != 17{
t.Error("AddTwoNums is failed",sum)
}
}
// AddTwoNums的测试用例2
// 测试AddTwoNums函数 结果错误
func TestAddTwoNums2(t *testing.T) {
sum := AddTwoNums(100,100)
if sum != 2000{
t.Error("AddTwoNums is failed",sum)
}
}
// AddUp的测试用例
// 测试AddUp函数
func TestAddUp(t *testing.T) {
sum := AddUp(100)
if sum != 5050{
t.Error("AddUp is failed",sum)
}
}
sham_test.go
package utils
import "testing"
func TestFunc1(t *testing.T) {
t.Log("TestFunc1 run")
}
func TestFunc2(t *testing.T) {
info := "TestFunc12 run"
t.Log(info + "===test log the method name `Log`")
t.Logf("output : %s", info+"===test log the method name `Logf`")
t.Error(info + "===test log the method name `Error`")
t.Errorf("output : %s", info+"===test log the method name `Errorf`")
t.Fatal(info + "===test log the method name `Fatal`")
t.Fatalf("output : %s", info+"===test log the method name `Fatalf`")
}
我们启动单元测试
使用的命令是 go test
PS:要查看这个命令这么使用 ,可以查看帮助文档 在命令行中执行
go help testflag
执行该目录下的所有测试用例
-v
显示详细的流程
go test -v
=== RUN TestAddTwoNums1
--- PASS: TestAddTwoNums1 (0.00s)
=== RUN TestAddTwoNums2
--- FAIL: TestAddTwoNums2 (0.00s)
func1_test.go:20: AddTwoNums is failed 200
=== RUN TestAddUp
--- PASS: TestAddUp (0.00s)
=== RUN TestFunc1
--- PASS: TestFunc1 (0.00s)
sham_test.go:6: TestFunc1 run
=== RUN TestFunc2
--- PASS: TestFunc2 (0.00s)
sham_test.go:9: TestFunc12 run
FAIL
exit status 1
FAIL GoNote/chapter10/demo3/utils 0.386s
执行指定的单元测试用例
-run
运行需要单独执行的测试用例
go test -v -run TestAddUp
=== RUN TestAddUp
--- PASS: TestAddUp (0.00s)
PASS
ok GoNote/chapter10/demo3/utils 0.392s
执行测试文件中的所有测试用例
go test -v sham.go
=== RUN TestFunc1
--- PASS: TestFunc1 (0.00s)
sham_test.go:6: TestFunc1 run
=== RUN TestFunc2
--- FAIL: TestFunc2 (0.00s)
sham_test.go:10: TestFunc12 run===test log the method name `Log`
sham_test.go:11: output : TestFunc12 run===test log the method name `Logf`
sham_test.go:12: TestFunc12 run===test log the method name `Error`
sham_test.go:13: output : TestFunc12 run===test log the method name `Errorf`
sham_test.go:14: TestFunc12 run===test log the method name `Fatal`
FAIL
FAIL command-line-arguments 0.383s
通过上面的例子,总结如下
- 测试用例文件名必须以
_test
结尾- 测试用例函数名必须以
Test
开头- 测试用例函数的形参必须是
*testing.T
类型- 单个测试文件中可以有多个测试用例
- 可以使用多个测试用例测试同一个内容
- 测试用例函数不需要放在
main
函数也能执行PASS
表示测试用例运行成功 ,FAIL
表示测试用例运行失败
单元测试框架提供的日志方法
方法名 | 含义 |
---|---|
Log | 打印日志 |
Logf | 格式化打印日志 |
Error | 打印错误日志, 将测试函数标记为失败,但是继续运行 |
Errorf | 格式化打印错误日志 ,将测试函数标记为失败,但是继续运行 |
Fatal | 打印致命错误日志,将测试函数标记为失败并停止执行 |
Fatalf | 格式化打印致命错误日志,将测试函数标记为失败并停止执行 |
2.1 提示 build failed
?
PS : 在执行测试文件的时候可能会出现类似下面的情况
$ go test func1_test.go # command-line-arguments [command-line-arguments.test] .\func1_test.go:10:9: undefined: AddTwoNums .\func1_test.go:19:9: undefined: AddTwoNums .\func1_test.go:28:9: undefined: AddUp FAIL command-line-arguments [build failed]
我们测试的两个函数
AddTwoNums
和AddUp
提示未定义, 最后提示命令行参数处理 构建失败
如何解决呢?
执行测试命令带上原文件即可
$ go test -v func1_test.go func1.go === RUN TestAddTwoNums1 --- PASS: TestAddTwoNums1 (0.00s) === RUN TestAddTwoNums2 --- FAIL: TestAddTwoNums2 (0.00s) func1_test.go:21: AddTwoNums is failed 200 === RUN TestAddUp --- PASS: TestAddUp (0.00s) FAIL FAIL command-line-arguments 0.336s
3. 基准测试
基准测可以测试一段程序的运行性能及耗费CPU的程度
写法上和测试用例的类似
package utils
import (
"fmt"
"testing"
)
// 基准测试 demo
func BenchmarkDemo1(b *testing.B) {
var n int
// b.N 是测试框架提供
for i := 0; i < b.N; i++ {
n++
}
}
func BenchmarkDemo2(b *testing.B) {
}
// 基准测试 测试函数AddTwoNums
func BenchmarkAddTwoNums(b *testing.B) {
for i := 0; i < b.N; i++ {
sum := AddTwoNums(i, i)
fmt.Sprintf("%d",sum)
}
}
执行测试文件
$ go test -v -bench=. -benchmem -benchtime=10s bm_test.go func1.go
goos: windows
goarch: amd64
BenchmarkDemo1-4 10000000000 0.28 ns/op 0 B/op 0 allocs/op
BenchmarkDemo2-4 10000000000 0.00 ns/op 0 B/op 0 allocs/op
BenchmarkAddTwoNums-4 100000000 112 ns/op 20 B/op 2 allocs/op
PASS
ok command-line-arguments 14.575s
命令说明
-bench=.
参数表示运行 bm_test.go 文件中所有的基准测试 ,如果想单独运行某个基准测试带上对应的名称即可,例如-bench=BenchmarkAddTwoNums
-benchmem
参数表示显示内存分配情况
-benchtime=
参数表示自定义测试时间测试结果说明
-4
表示4个CPU线程执行
ns/op
表示每一个操作消耗多少时间,单位是 纳秒ns
B/op
表示每一次操作需要分配的字节数
allocs/op
表示每次执行分配了多少次