开发中需要确认一个函数、模块结果是否正确
func addUpper(n int)int{ | |
res:=0for i := 1; i <n ; i++ { | |
res+=i | |
}return res | |
} |
传统方法测试就是调用这个函数,看返回结果是否正确
缺点:
- 需要在main函数中调用,测试的时候去修改main函数,运行中的项目得停止来测试
- 不利于管理,因为当我们测试多个函数或者多个模块时,都要写在main函数中,逻辑混乱
- 引出党员测试,testing测试框架,可以很好解决问题
基本介绍
Go 语言中自带有一个轻量级的测试框架 testing 和自带的 go test 命令来实现单元测试和性能测试,testing 框架可以写测试用例、压力测试用例,
单元测试作用 :
- 确保每个函数是可运行,并且运行结果是正确的
- 确保写出来的代码性能是好的,
- 单元测试能及时的发现程序设计或实现的逻辑错误,使问题及早暴露,便于问题的定位解决, 而性能测试的重点在于发现程序设计上的一些问题,让程序能够在高并发的情况下还能保持稳定
单元测试
cal.go
func AddUpper(n int)int{ | |
res:=0for i := 0; i < n; i++ { | |
res += i | |
}return res | |
} |
cal_test.go
//编写测试用例,测试addUpper是否正常 | |
func TestAddUpper(t *testing.T) { | |
res := AddUpper(10)if res != 55 { | |
fmt.Println("AddUpper(10)执行错误,期望值=%v 实际值=%v", 55, res)} | |
//输出日志 | |
t.Logf("AddUpper(10)执行正确") | |
} | |
C:\Users\55480\go\go\learn\testcase01>go test -v | |
# learngo/learn/testcase01 | |
.\cat_test.go:15:3: Println call has possible formatting directive %v | |
FAIL learngo/learn/testcase01 [build failed] |
# learngo/learn/testcase01 | |
.\cat_test.go:15:3: Println call has possible formatting directive %v | |
FAIL learngo/learn/testcase01 [build failed] | |
C:\Users\55480\go\go\learn\testcase01>go test -v | |
=== RUN TestAddUpper | |
AddUpper(10)执行错误,期望值=55 实际值=45 cat_test.go:18: AddUpper(10)执行正确 | |
--- PASS: TestAddUpper (0.00s) | |
PASS | |
ok learngo/learn/testcase01 0.080s |
testing框架,会将xxx_test.go的文件加入,调用testXxx方法
总结
- 测试用例文件名必须以_test.go结尾
- 测试用例函数必须以Test开头(Test+函数名)
- 形参必须是(T *tesing.T)
- 一个测试用例文件中,可以有多个测试用例函数,也可与在不同文件,只要命名统一
- 运行测试用例指令
- go test 运行正确,无日志,错误会输出错误日志
- go test -v 正确错误都会输出日志
- 当出现错误时,可以使用 t.Fatalf 来格式化输出错误信息,并退出程序
- t.Logf 方法可以输出相应的日志
- 测试单个文件,一定要带上被测试的原文件 :
C:\Users\55480\go\learn\testcase01>go test -v cat_test.go cal.go
- 测试单个方法 :
go test -v -test.run TestAddUpper
- 默认扫描当前文件夹内所有的测试用例
案例:
package main | |
import ("encoding/json""fmt""io/ioutil" | |
) | |
type monster struct { | |
Name string | |
Age int | |
Skill string | |
} | |
//给monster绑定方法store,可以将一个monster变量(对象),序列化后保存到文件中 | |
func (this *monster) Store()bool{//先序列化 | |
data,err:=json.Marshal(this)if err!=nil{ | |
fmt.Println("marshal err=",err)return false} | |
//保存到文件 | |
filePath:="c/monster.ser" | |
err=ioutil.WriteFile(filePath,data,0777)if err!=nil{ | |
fmt.Println("write fiule err=",err)return false}return true | |
} | |
func (this *monster) ReStore()bool{//反先序列化 | |
filePath:="d/monster.ser" | |
data,err:=ioutil.ReadFile(filePath)if err!=nil{ | |
fmt.Println("readFile ",err)return false} | |
//反序列化 | |
err= json.Unmarshal(data,this)if err!=nil{ | |
fmt.Println("Unmarshal err ",err)return false}return true | |
} |
测试案例
package main | |
import "testing" | |
func TestStore(t *testing.T){//先创建一个monster实例 | |
monster := &monster{ | |
Name:"擦拭", | |
Age:12, | |
Skill:"风火轮",} | |
res := monster.Store()if res{ | |
t.Fatalf("方法错误,希望为%v,实际为%v",true,res)} | |
t.Logf("测试成功") | |
} | |
func TestReStore(t *testing.T){//先创建一个monster实例var monster= &monster{} | |
res:=monster.ReStore()if res{ | |
t.Fatalf("方法错误,希望为%v,实际为%v",true,res)} | |
if monster.Name !="擦拭" { | |
t.Fatalf("方法错误,希望为%v,实际为%v","擦拭",res) | |
} | |
t.Logf("测试成功") | |
} |