go testing包支持开箱即用的基准测试,需用go test -bench运行以BenchmarkXxx命名的函数;函数名须以Benchmark开头、参数为*testing.B、循环使用b.N。

Go 自带的 testing 包提供了开箱即用的基准测试(benchmark)能力,无需额外依赖。核心是用 go test -bench 运行以 BenchmarkXxx 命名的函数,Go 会自动执行多次并统计平均耗时、内存分配等关键指标。
写一个合法的 Benchmark 函数
基准测试函数必须满足三个条件:
示例:
func BenchmarkConcatString(b *testing.B) {
for i := 0; i _ = “hello” + “world”
}
}
运行和解读 benchmark 结果
在包目录下执行:
立即学习“go语言免费学习笔记(深入)”;
go test -bench=.
输出类似:
BenchmarkConcatString-8 1000000000 0.32 ns/op
含义:
-
BenchmarkConcatString-8:函数名 + CPU 逻辑核数(-8 表示 8 核) -
1000000000:实际运行次数(b.N 自适应调整,确保测试时间足够稳定) -
0.32 ns/op:每次操作平均耗时 0.32 纳秒
加 -benchmem 可查看内存分配情况:go test -bench=. -benchmem
避免常见陷阱,让结果更真实
基准测试容易因编译器优化或干扰逻辑失真,需主动规避:
- 防止死代码消除:如果结果没被使用,编译器可能直接优化掉整段逻辑。用
b.ReportAllocs()或将结果赋给全局变量(如result = xxx),或用blackhole方式:blackhole = result(var blackhole Interface{}) - 避免在循环外做初始化:如 map 初始化、切片预分配等,应放在
b.ResetTimer()之后,否则会混入准备时间 - 慎用
b.StopTimer()/b.StartTimer():仅在必须排除 setup/teardown 开销时使用,比如读文件、生成测试数据
对比多个实现,用子基准测试组织
用 b.Run 分组测试不同方案,结构清晰且可单独运行:
func BenchmarkStringMethods(b *testing.B) {
b.Run(“concat”, func(b *testing.B) {
for i := 0; i _ = “a” + “b”
}
})
b.Run(“sprintf”, func(b *testing.B) {
for i := 0; i _ = fmt.Sprintf(“%s%s”, “a”, “b”)
}
})
}
运行指定子项:go test -bench=StringMethods/concat
基本上就这些。Go 的 benchmark 简洁但严谨,关键是写对函数签名、理解 b.N 机制、避开优化陷阱。测准了,才能真正看出 slice 预分配、sync.Pool、unsafe 转换这些优化是否值得。