十五、单元测试 
                                                    
                        
                    
                    
  
                    
                    开发中需要确认一个函数、模块结果是否正确
func addUpper(n int)int{
   res:=0
   for 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:=0
   for 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("测试成功")
}
本作品采用《CC 协议》,转载必须注明作者和本文链接
 
           CR 的个人博客
 CR 的个人博客
         
             
           
           关于 LearnKu
                关于 LearnKu
               
                     
                     
                     粤公网安备 44030502004330号
 粤公网安备 44030502004330号 
 
推荐文章: