首先安装单元测试包,go get github.com/smartystreets/goconvey/convey

源程序如下,定义了加减乘除4个函数

package test222

import (
"errors"
)

func Add(a, b int) int {
return a + b
}

func Subtract(a, b int) int {
return a - b
}

func Multiply(a, b int) int {
return a * b
}

func Division(a, b int) (int, error) {
if b == 0 {
return 0, errors.New("被除数不能为 0")
}
return a / b, nil
}
 测试程序如下

package test222

import (
"testing"

. "github.com/smartystreets/goconvey/convey"
)

func TestAdd(t *testing.T) {
Convey("将两数相加", t, func() {
So(Add(1, 2), ShouldEqual, 4)
})
}

func TestSubtract(t *testing.T) {
Convey("将两数相减", t, func() {
So(Subtract(1, 2), ShouldEqual, -1)
})
}

func TestMultiply(t *testing.T) {
Convey("将两数相乘", t, func() {
So(Multiply(3, 2), ShouldEqual, 6)
})
}

func TestDivision(t *testing.T) {
Convey("将两数相除", t, func() {

Convey("除以非 0 数", func() {
num, err := Division(10, 2)
So(err, ShouldBeNil)
So(num, ShouldEqual, 5)
})

Convey("除以 0", func() {
_, err := Division(10, 0)
So(err, ShouldNotBeNil)
})
})
}
测试文件的格式有几点要求:

1  测试程序的package名必须和源程序相同

2  文件名必须是*_test.go的类型,*代表要测试的文件名

3  函数名必须以Test开头如:TestXxx或Test_xxx

4  测试函数TestXxx()的参数是*testing.T

5 测试内容都包含在Convey函数中

测试方法:

1  测试单个文件

go test -v  func_test.go func.go

2  测试单个方法

go test -v -test.run TestAdd
---------------------

性能测试系统可以给出代码的性能数据,帮助测试者分析性能问题。

提示

单元测试(unit testing),是指对软件中的最小可测试单元进行检查和验证。对于单元测试中单元的含义,一般要根据实际情况去判定其具体含义,如C语言中单元指一个函数,Java 里单元指一个类,图形化的软件中可以指一个窗口或一个菜单等。总的来说,单元就是人为规定的最小的被测功能模块。

单元测试是在软件开发过程中要进行的最低级别的测试活动,软件的独立单元将在与程序的其他部分相隔离的情况下进行测试。

单元测试——测试和验证代码的框架

要开始一个单元测试,需要准备一个 go 源码文件,在命名文件时需要让文件必须以_test结尾。

单元测试源码文件可以由多个测试用例组成,每个测试用例函数需要以Test为前缀,例如:

func TestXXX( t *testing.T )

  • 测试用例文件不会参与正常源码编译,不会被包含到可执行文件中。
  • 测试用例文件使用 go test 指令来执行,没有也不需要 main() 作为函数入口。所有在以_test结尾的源码内以Test开头的函数会自动被执行。
  • 测试用例可以不传入 *testing.T 参数。

helloworld 的测试代码(具体位置是./src/chapter11/gotest/helloworld_test.go):

本套教程所有源码下载地址:https://pan.baidu.com/s/1ORFVTOLEYYqDhRzeq0zIiQ    提取密码:hfyf

  1. package code11_3
  2. import "testing"
  3. func TestHelloWorld(t *testing.T) {
  4. t.Log("hello world")
  5. }

代码说明如下:

  • 第 5 行,单元测试文件 (*_test.go) 里的测试入口必须以 Test 开始,参数为 *testing.T 的函数。一个单元测试文件可以有多个测试入口。
  • 第 6 行,使用 testing 包的 T 结构提供的 Log() 方法打印字符串。

1) 单元测试命令行

单元测试使用 go test 命令启动,例如:

$ go test helloworld_test.go
ok          command-line-arguments        0.003s
$ go test -v helloworld_test.go
=== RUN   TestHelloWorld
--- PASS: TestHelloWorld (0.00s)helloworld_test.go:8: hello world
PASS
ok          command-line-arguments        0.004s

代码说明如下:

  • 第 1 行,在 go test 后跟 helloworld_test.go 文件,表示测试这个文件里的所有测试用例。
  • 第 2 行,显示测试结果,ok 表示测试通过,command-line-arguments 是测试用例需要用到的一个包名,0.003s 表示测试花费的时间。
  • 第 3 行,显示在附加参数中添加了-v,可以让测试时显示详细的流程。
  • 第 4 行,表示开始运行名叫 TestHelloWorld 的测试用例。
  • 第 5 行,表示已经运行完 TestHelloWorld 的测试用例,PASS 表示测试成功。
  • 第 6 行打印字符串 hello world。

2) 运行指定单元测试用例

go test 指定文件时默认执行文件内的所有测试用例。可以使用-run参数选择需要的测试用例单独执行,参考下面的代码。

一个文件包含多个测试用例(具体位置是./src/chapter11/gotest/select_test.go

  1. package code11_3
  2. import "testing"
  3. func TestA(t *testing.T) {
  4. t.Log("A")
  5. }
  6. func TestAK(t *testing.T) {
  7. t.Log("AK")
  8. }
  9. func TestB(t *testing.T) {
  10. t.Log("B")
  11. }
  12. func TestC(t *testing.T) {
  13. t.Log("C")
  14. }

这里指定 TestA 进行测试:

$ go test -v -run TestA select_test.go
=== RUN   TestA
--- PASS: TestA (0.00s)select_test.go:6: A
=== RUN   TestAK
--- PASS: TestAK (0.00s)select_test.go:10: AK
PASS
ok          command-line-arguments        0.003s

TestA 和 TestAK 的测试用例都被执行,原因是-run跟随的测试用例的名称支持正则表达式,使用-run TestA$即可只执行 TestA 测试用例。

3) 标记单元测试结果

当需要终止当前测试用例时,可以使用 FailNow,参考下面的代码。

测试结果标记(具体位置是./src/chapter11/gotest/fail_test.go

  1. func TestFailNow(t *testing.T) {
  2. t.FailNow()
  3. }

还有一种只标记错误不终止测试的方法,代码如下:

  1. func TestFail(t *testing.T) {
  2. fmt.Println("before fail")
  3. t.Fail()
  4. fmt.Println("after fail")
  5. }

测试结果如下:

=== RUN   TestFail
before fail
after fail
--- FAIL: TestFail (0.00s)
FAIL
exit status 1
FAIL        command-line-arguments        0.002s

从日志中看出,第 5 行调用 Fail() 后测试结果标记为失败,但是第 7 行依然被程序执行了。

4) 单元测试日志

每个测试用例可能并发执行,使用 testing.T 提供的日志输出可以保证日志跟随这个测试上下文一起打印输出。testing.T 提供了几种日志输出方法,详见下表所示。

单元测试框架提供的日志方法
方  法 备  注
Log 打印日志,同时结束测试
Logf 格式化打印日志,同时结束测试
Error 打印错误日志,同时结束测试
Errorf 格式化打印错误日志,同时结束测试
Fatal 打印致命日志,同时结束测试
Fatalf 格式化打印致命日志,同时结束测试

开发者可以根据实际需要选择合适的日志。

基准测试——获得代码内存占用和运行效率的性能数据

基准测试可以测试一段程序的运行性能及耗费 CPU 的程度。Go 语言中提供了基准测试框架,使用方法类似于单元测试,使用者无须准备高精度的计时器和各种分析工具,基准测试本身即可以打印出非常标准的测试报告。

1) 基础测试基本使用

下面通过一个例子来了解基准测试的基本使用方法。

基准测试(具体位置是./src/chapter11/gotest/benchmark_test.go

  1. package code11_3
  2. import "testing"
  3. func Benchmark_Add(b *testing.B) {
  4. var n int
  5. for i := 0; i < b.N; i++ {
  6. n++
  7. }
  8. }

这段代码使用基准测试框架测试加法性能。第 7 行中的 b.N 由基准测试框架提供。测试代码需要保证函数可重入性及无状态,也就是说,测试代码不使用全局变量等带有记忆性质的数据结构。避免多次运行同一段代码时的环境不一致,不能假设 N 值范围。

使用如下命令行开启基准测试:

$ go test -v -bench=. benchmark_test.go
goos: linux
goarch: amd64
Benchmark_Add-4           20000000         0.33 ns/op
PASS
ok          command-line-arguments        0.700s

代码说明如下:

  • 第 1 行的-bench=.表示运行 benchmark_test.go 文件里的所有基准测试,和单元测试中的-run类似。
  • 第 4 行中显示基准测试名称,2000000000 表示测试的次数,也就是 testing.B 结构中提供给程序使用的 N。“0.33 ns/op”表示每一个操作耗费多少时间(纳秒)。

注意:Windows 下使用 go test 命令行时,-bench=.应写为-bench="."

2) 基准测试原理

基准测试框架对一个测试用例的默认测试时间是 1 秒。开始测试时,当以 Benchmark 开头的基准测试用例函数返回时还不到 1 秒,那么 testing.B 中的 N 值将按 1、2、5、10、20、50……递增,同时以递增后的值重新调用基准测试用例函数。

3) 自定义测试时间

通过-benchtime参数可以自定义测试时间,例如:

$ go test -v -bench=. -benchtime=5s benchmark_test.go
goos: linux
goarch: amd64
Benchmark_Add-4           10000000000                 0.33 ns/op
PASS
ok          command-line-arguments        3.380s

4) 测试内存

基准测试可以对一段代码可能存在的内存分配进行统计,下面是一段使用字符串格式化的函数,内部会进行一些分配操作。

  1. func Benchmark_Alloc(b *testing.B) {
  2. for i := 0; i < b.N; i++ {
  3. fmt.Sprintf("%d", i)
  4. }
  5. }

在命令行中添加-benchmem参数以显示内存分配情况,参见下面的指令:

$ go test -v -bench=Alloc -benchmem benchmark_test.go
goos: linux
goarch: amd64
Benchmark_Alloc-4 20000000 109 ns/op 16 B/op 2 allocs/op
PASS
ok          command-line-arguments        2.311s

代码说明如下:

  • 第 1 行的代码中-bench后添加了 Alloc,指定只测试 Benchmark_Alloc() 函数。
  • 第 4 行代码的“16 B/op”表示每一次调用需要分配 16 个字节,“2 allocs/op”表示每一次调用有两次分配。

开发者根据这些信息可以迅速找到可能的分配点,进行优化和调整。

5) 控制计时器

有些测试需要一定的启动和初始化时间,如果从 Benchmark() 函数开始计时会很大程度上影响测试结果的精准性。testing.B 提供了一系列的方法可以方便地控制计时器,从而让计时器只在需要的区间进行测试。我们通过下面的代码来了解计时器的控制。

基准测试中的计时器控制(具体位置是./src/chapter11/gotest/benchmark_test.go):

  1. func Benchmark_Add_TimerControl(b *testing.B) {
  2. // 重置计时器
  3. b.ResetTimer()
  4. // 停止计时器
  5. b.StopTimer()
  6. // 开始计时器
  7. b.StartTimer()
  8. var n int
  9. for i := 0; i < b.N; i++ {
  10. n++
  11. }
  12. }

从 Benchmark() 函数开始,Timer 就开始计数。StopTimer() 可以停止这个计数过程,做一些耗时的操作,通过 StartTimer() 重新开始计时。ResetTimer() 可以重置计数器的数据。

计数器内部不仅包含耗时数据,还包括内存分配的数据。

http://c.biancheng.net/golang/struct/

go test 单元函数测试相关推荐

  1. pdfbox 按章节读取_2020年智慧树APP微生物与健康第五单元章节测试网课答案大学课后答案...

    2020年智慧树APP微生物与健康第五单元章节测试网课答案大学课后答案 更多相关问题 [多选题]<人性的弱点>孔子说:"见贤思齐焉,见不贤而内自省也".当你读到这句话时 ...

  2. GO(单元)测试的基本规则和流程

    GO(单元)测试的基本规则和流程 文章目录 GO(单元)测试的基本规则和流程 一.GO程序的三类单元测试 二.GO语言对测试函数的名称和签名的规定 三.`go test`命令执行的主要测试流程 四.功 ...

  3. exfat 分配单元大小_知到金融理论与实务第一单元章节测试答案

    知到金融理论与实务第一单元章节测试答案 更多相关问题 根据图4-18所示顺序阀结构原理图,试问: 1) 如将调压弹簧腔的外泄油口安装成内部回油形式,即L口与p2口相 对于企业取得的无形资产,应当自可供 ...

  4. 经典笔试题:用C写一个函数测试当前机器大小端模式

    "用C语言写一个函数测试当前机器的大小端模式"是一个经典的笔试题,如下使用两种方式进行解答: 1. 用union来测试机器的大小端 1 #include <stdio.h&g ...

  5. pdfbox 按章节读取_2020年知到设施园艺学第十单元章节测试答案

    2020年知到设施园艺学第十单元章节测试答案 更多相关问题 [判断题] 转子流量计的压力损失大,并且随流量大小而变化. [判断题] 在超声波流量计中,小管径传感器的声音发射频率高,大管径传感器中的声音 ...

  6. Boost:用成员函数测试bind <void>

    Boost:用成员函数测试bind 实现功能 C++实现代码 实现功能 boost::bind模块,用成员函数测试bind C++实现代码 #include <boost/config.hpp& ...

  7. 2020知到python语言应用答案_2020年知到APPPython语言应用第四单元章节测试答案

    2020年知到APPPython语言应用第四单元章节测试答案 更多相关问题 一般情况下,协商者要把自己所设定的目标划分为()A.理想目标B.可接受目标C.最低目标D.最高目标 银行在电子商务中扮演的角 ...

  8. wnoise matlab,MATLAB中用wnoise函数测试去噪算法

    MATLAB中用wnoise函数测试去噪算法 sqrt_snr=3; init=231434; [x,xn]=wnoise(3,11,sqrt_snr,init); % WNOISE generate ...

  9. python123实验七答案_2020年智慧树嵌入式系统与实验第七单元章节测试答案

    2020年智慧树嵌入式系统与实验第七单元章节测试答案 更多相关问题 [多选题]含挥发油丰富,易散失气味的中药饮片有() A. 荆芥 B. 薄荷 C. 当归 D. 白芷 E. 肉桂 [单选题]求大同存小 ...

最新文章

  1. (邓爱萍)继承 课本
  2. 【C 语言】文件操作 ( 配置文件读写 | 框架搭建 | 头文件定义 | 头文件导入限制 | 兼容 C++ 语言 | 函数形参输入输出属性注释)
  3. Swift - 30 - 可变参数
  4. 【Python】itertools之product函数
  5. linux 查看mysql运行时间_linux – strace显示从mysql socket读取很长时间 – mysql需要很长时间才能执行查询?...
  6. 教你在Excel里做GA的水平百分比图的详细步骤(图文教程)-成为excel大师(1)...
  7. 硬盘序列号示例_序列化代理模式示例
  8. Python使用Tkinter实现转盘抽奖器
  9. std::map的使用
  10. 实用的 Python 之 feedparser
  11. oracle性能优化求生指南_Vue项目性能优化--实践指南,网上最全最详细
  12. java中的math pow,Java Math.pow(a,b)时间复杂度
  13. SNAT、DNAT、MASQUERADE的区别
  14. Web前端开发有哪些岗位职责 要掌握什么技能
  15. 代码何须全部自己写,融云守护程序猿发际线
  16. 那些年啊,那些事——一个程序员的奋斗史 ——83
  17. 计算机bios设置系统安装教程,z590主板装win7系统及bios设置教程(支持11代cpu驱动)...
  18. C# DGV常用操作
  19. Matlab上位机开发(一) —— 了解上位机
  20. java编程那些事儿 pdf,科学网—《java编程那些事儿》 前面部分阅读笔记 - 马舒天的博文...

热门文章

  1. 欢迎报名广东省教育厅2022年科技劳动教育实践活动
  2. 【通信原理】信道编码——线性分组码
  3. 密码正确登入不了网络计算机,电脑突然断网,然后重启输入密码就登陆不了电脑为什么?老是显示当前电脑时间与网络时间不一致...
  4. C语言学习笔记 |进阶| 内存函数
  5. 【CSS】PhotoShop 切图 ② ( PhotoShop 切片选择工具 | 清除切片 | 新建基于图层的切片 | 透明背景图片切图 | 根据参考线选择切片 )
  6. Solidity 最新 0.5.8 中文文档发布
  7. 数字电路 模拟电路 先学哪个_数字电路,模拟电路
  8. BLAG: Improving the Accuracy of Blacklists
  9. html百度地图取电,盯盯拍支持高德地图AR导航,升级智慧出行新体验
  10. 低代码开发平台到底是何方圣神? 居然可以虏获世界500强等企业的芳心!