Hello,各位小伙伴,我们今天来学习一下go的测试,希望给各位一点点小小的帮助
有过编程基础的,或者工作的了,应该知道,一个软件的质量,三分靠开发,七分靠测试。如果没有及时发现存在的问题,将会给这个公司带来巨大的经济损失。老哥因为转Go了,所以非常有必要学习一个Go 相关的测试。
话不多说,开始学习吧。
1、在go语言中,一共有4种测试,参考文档,https://pkg.go.dev/testing
类型 | 格式 | 作用 | 传入参数 |
---|---|---|---|
单元测试 | 函数的前缀名为Test | 一个函数正常测试功能 | t *testing.T |
基准(压力测试) | 函数的前缀名为Benchmark | 通过cpu和内存,并发测试程序的性能 | b *testing.B |
实例测试 | 函数的前缀名为Example | 给测试人员提供实例文档 | m *testing.M |
模糊(随机) 测试 | 函数的前缀名为Fuzz | 生成一个随机测试用例去覆盖可能人为测不到的地方 | f *testing.F |
单元测试calc.go
package main
func Add(a int, b int) int {
return a + b
}
func Mul(a int, b int) int {
return a * b
}
calc_test.go
package main
func TestAdd(t *testing.T) {
if value := Add(1, 2); value != 3 {
t.Errorf("1+2 expected be 3, but %d got", value)
}
if value := Mul(1, 2); value != 2 {
t.Errorf("1*2 expected be 2, but %d got", value)
}
}
点击这个绿色图标,就可以开始测试了
除了点击绿色图标以外,我们也可以换一种方式,到当前文件夹下,打开Goland控制终端,输入下面的命令,它会执行当前文件夹所有的测试文件,并且输出详细的信息
PS D:\Project\Go_Project\testCobraPlus> go test -v
3A、单元测试如果有多个用例的话,可以试试嵌套测试,它可以分开测试
// 嵌套测试
func TestMul(t *testing.T) {
t.Run("pos", func(t *testing.T) {
if Mul(2, 3) != 6 {
t.Fatal("expected to get 6,but fail...")
}
})
t.Run("neg", func(t *testing.T) {
if Mul(2, -3) != -6 {
t.Fatal("expected to get -6,but fail...")
}
})
}
点击第一个,会测试全部用例,点击第二个或者第三个,它会测试你点击的那一个
5、单元测试第三种形式,也是k8s用的测试形式,它这种用要给切片存放多条数据,可以一次性测试多个值,推荐使用
func TestMul2(t *testing.T) {
cases := []struct {
Name string
A, B, Expected int
}{
{"pos", 2, 3, 6},
{"neg", 2, -3, -6},
{"zero", 2, 0, 0},
}
for _, c := range cases {
t.Run(c.Name, func(t *testing.T) {
if value := Mul(c.A, c.B); value != c.Expected {
t.Fatalf("%d * %d expected %d,but %d got", c.A, c.B, c.Expected, value)
}
})
}
}
6、单元测试钩子,如果我们加了以下几行代码,那么当我们点击了3、3A、5任何一个测试用例,以下代码就会执行
func before() {
fmt.Println("Before all tests...")
}
func after() {
fmt.Println("After all tests...")
}
func TestMain(m *testing.M) {
before()
code := m.Run()
after()
os.Exit(code)
}
network_test.go
package main
import (
"io"
"net/http"
"net/http/httptest"
"testing"
)
func helloHandler(w http.ResponseWriter, r *http.Request) {
w.Write([]byte("hello baidu"))
}
func TestConn(t *testing.T) {
req := httptest.NewRequest("GET", "https://www.baidu.com/", nil)
w := httptest.NewRecorder()
helloHandler(w, req)
bytes, _ := io.ReadAll(w.Result().Body)
if string(bytes) != "hello baidu" {
t.Fatal("expected hello baidu,but got", string(bytes))
}
}
8、然后是基准(压力)测试,这里需要说明一下,压力测试和单元测试最大的区别是单元测试它只执行一次,成功一次就算成功,而压力测试需要执行多次,任何一次不成功都算失败
package main
import (
"bytes"
"testing"
)
func Benchmark_Add(b *testing.B) {
var n int
//b.N 是基准测试框架提供的,表示循环的次数,没有具体的限制
for i := 0; i < b.N; i++ {
n++
}
}
func BenchmarkConcatStringByAdd(b *testing.B) {
//有些测试需要一定的启动和初始化时间,如果从 Benchmark() 函数开始计时会很大程度上影响测试结果的精准性。
//testing.B 提供了一系列的方法可以方便地控制计时器,从而让计时器只在需要的区间进行测试
elem := []string{"1", "2", "3", "4", "5"}
b.ResetTimer()
for i := 0; i < b.N; i++ {
ret := ""
for _, v := range elem {
ret += v
}
}
b.StopTimer()
}
func BenchmarkConcatStringByByteBuffer(b *testing.B) {
elems := []string{"1", "2", "3", "4", "5"}
b.ResetTimer()
for i := 0; i < b.N; i++ {
var buf bytes.Buffer
for _, elem := range elems {
buf.WriteString(elem)
}
}
b.StopTimer()
}
同样,我们也可以在goland控制终端,查看执行信息,终端操作会执行当前文件夹的所有压力测试用例
goos: windows goarch: amd64 cpu: 12th Gen Intel(R) Core(TM) i7-12700KF
Benchmark_Add-20 1000000000 0.1053 ns/op
PS D:\Project\Go_Project\testCobraPlus\benchmark> go test -v -bench="." benchmark_test.go
goos: windows
goarch: amd64
cpu: 12th Gen Intel(R) Core(TM) i7-12700KF
Benchmark_Add
Benchmark_Add-20 1000000000 0.1053 ns/op
BenchmarkConcatStringByAdd
BenchmarkConcatStringByAdd-20 20758946 57.88 ns/op
BenchmarkConcatStringByByteBuffer
BenchmarkConcatStringByByteBuffer-20 40134450 28.70 ns/op
PASS
ok command-line-arguments 2.694s
PS D:\Project\Go_Project\testCobraPlus\benchmark>
9、压力测试自定义测试时间,也就是测试多少秒,默认是测试1秒的压测情况
PS D:\Project\Go_Project\testCobraPlus\benchmark> go test -v -bench="." -benchtime=5s benchmark_test.go
goos: windows
goarch: amd64
cpu: 12th Gen Intel(R) Core(TM) i7-12700KF
Benchmark_Add
Benchmark_Add-20 1000000000 0.1051 ns/op
BenchmarkConcatStringByAdd
BenchmarkConcatStringByAdd-20 98688592 58.19 ns/op
BenchmarkConcatStringByByteBuffer
BenchmarkConcatStringByByteBuffer-20 214375582 28.14 ns/op
PASS
ok command-line-arguments 14.921s
10、压力测试查看内存分配了多少,如
-bench=Add 指定尾缀是Add方法,都会执行
-benchmem 查看内存分配多少,比如下面一个方法,16 B/op 表示一次调用需要分配16个字节, 4 allocs/op 表示每一次调用有4次分配
PS D:\Project\Go_Project\testCobraPlus\benchmark> go test -v -bench=Add -benchmem benchmark_test.go
goos: windows
goarch: amd64
cpu: 12th Gen Intel(R) Core(TM) i7-12700KF
Benchmark_Add
Benchmark_Add-20 1000000000 0.1060 ns/op 0 B/op
0 allocs/op
BenchmarkConcatStringByAdd
BenchmarkConcatStringByAdd-20 19738010 57.19 ns/op 16 B/op
4 allocs/op
PASS
ok command-line-arguments 1.439s
11、实例测试,这个测试有点简单,写个 //Output注释,然后后面是期望输出的结果,如果是一致的,测试通过,否则测试失败
func Hello() string {
return "Hello World"
}
//Output 需要和上面打印的内容一致,否则测试失败
func ExampleHello() {
fmt.Println(Hello())
// Output: Hello World
}
func TestMain(m *testing.M) {
fmt.Println("Before...")
code := m.Run()
fmt.Println("End...")
os.Exit(code)
}
和上面一样,我们也可以在控制台终端,进行实例测试
PS D:\Project\Go_Project\testCobraPlus\benchmark> go test -v -run="ExampleHello"
Before...
=== RUN ExampleHello
--- PASS: ExampleHello (0.00s)
PASS
End...
ok testCobraPlus/benchmark 0.125s
12、模糊测试,这是go1.18新推出的测试用例,如果我们没有指定参数,系统会随机填入参数进行测试
func FuzzStrToNum(f *testing.F) {
f.Fuzz(func(t *testing.T, a string) {
b, _ := strconv.ParseInt(a, 10, 64)
fmt.Printf("%d\n", b)
})
}
func FuzzDivision(f *testing.F) {
f.Fuzz(func(t *testing.T, a, b int) {
fmt.Println(a / b)
})
}
//我们可以在模糊测试中,自己定义参数进去
func FuzzHello(f *testing.F) {
f.Add(1)
f.Fuzz(func(t *testing.T, num int) {
if num != 6 {
t.Errorf("expected 6,but got %d", num)
}
})
}
模糊测试同样支持控制台输入,但是同一个文件只能有一个模糊函数测试
PS D:\Project\Go_Project\testCobraPlus\benchmark> go test -fuzztime 10s -fuzz .
13、最后,各位小伙伴,麻烦给老哥一个点赞、关注、收藏三连好吗,你的支持是老哥更新最大的动力,谢谢!