单元测试
类型 | 前缀 | 签名 | 用途 |
---|---|---|---|
测试函数 | Test | func TestXxx(t *testing.T) | 功能测试、验证逻辑正确性 |
基准函数 | Benchmark | func BenchmarkXxx(b *testing.B) | 性能测试、效率评估 |
示例函数 | Example | func ExampleXxx() | 用法展示、生成文档 |
testing框架
- 文件名以_test.go结尾,放在与被测试包相同的包中
- 测试用例函数名以Test开头,一般为Test+被测试的函数名)
- **TestaddUpper(*testing.T)**的形参类型必须是*testing.T
- 一个测试用例文件中,可以有很多测试用例函数,比如TestAddUpper、TestSub
- 运行测试用例指令
- cmd>go test [运行正确,无日志;运行错误,输出日志]
- cmd>go test -v [运行正确或是错误,都将输出日志]
- t.Fatalf方法可以格式化输出错误信息,并退出程序
- t.Logf方法可以输出相应日志
- PASS表示运行成功,FAIL表示运行失败
func addUpper(n int) int {res := 0for i := 1; i <= n; i++ {res += i}return res
}func TestAddUpper(t *testing.T) {res := addUpper(10)if res != 55 {//fmt.Println("AddUpper(10) 执行错误,期望值=%v 实际值=%v", 55, res)t.Fatalf("AddUpper(10) 执行错误,期望值=%v 实际值=%v", 55, res)}//如果正确,输出日志t.Logf("AddUpper(10) 执行正确...")
}
实例:
// 给Monster绑定方法Store,可以将一个Monster变量(对象),序列化后保存到文件中
func (this *Monster) Store() bool {//先序列化data, err := json.Marshal(this)if err != nil {fmt.Println("marshal err =", err)return false}//保存到文件filePath := "D:\\GoLand-go\\filedemo.txt"err = ioutil.WriteFile(filePath, data, 0666)if err != nil {fmt.Println("write file err =", err)return false}return true
}// 给Monster绑定方法ReStore,可以将一个序列化的Monster,从文件中读取
// 并且反序列化为monster对象,检查反序列化,名字正确
func (this *Monster) ReStore() bool {//先从文件中,读取序列化的字符串filePath := "D:\\GoLand-go\\filedemo.txt"data, err := ioutil.ReadFile(filePath)if err != nil {fmt.Println("ReadFile err =", err)return false}//使用读取到data []byte,对反序列化err = json.Unmarshal(data, this)if err != nil {fmt.Println("UnMarshal err = ", err)return false}return true
}// 测试用例,测试Store方法
func TestMonster_Store(t *testing.T) {//先创建一个Monster实例monster := Monster{Name: "红孩儿",Age: 10,Skill: "吐火",}res := monster.Store()if !res {t.Fatalf("monster.Store() 错误,希望为=%v 实际为=%v", true, res)}t.Logf("monster.Store() 测试成功!")
}// 测试用例,测试ReStore方法
func TestMonster_ReStore(t *testing.T) {//先创建一个Monster实例,不需要指定字段的值var monster Monsterres := monster.ReStore()if !res {t.Fatalf("monster.ReStore() 错误,希望为=%v 实际为=%v", true, res)}//进一步判断if monster.Name != "红孩儿" {t.Fatalf("monster.ReStore() 错误,希望为=%v 实际为=%v", "红孩儿", monster.Name)}t.Logf("monster.ReStore() 测试成功!")
}
基准测试
基准测试以Benchmark为前缀,需要一个testing.B类型的参数b*,基准测试必须要执行b.N次,这样的测试才有对照性,b.N的值是系统根据实际情况去调整的,从而保证测试的稳定性
基准测试并不会默认执行,需要增加**-bench参数(还可以为基准测试添加-benchmem**参数,来获得内存分配的统计数据)
默认情况下,每个基准测试至少运行1秒。如果在Benchmark函数返回时没有到1秒,则b.N的值会按1,2,5,10,20,50,…增加,并且函数再次运行
b.ResetTimer之前的处理不会放到执行时间里,也不会输出到报告中,所以可以在之前做一些不计划作为测试报告的操作