Go测试需重视组织方式、执行策略与边界处理:用testing.Short()区分单元与集成测试,表格驱动降低维护成本,基准测试须满足结果使用、初始化剥离和多次运行三条件,集成测试需严格清理资源。
go test 本身足够轻量,但写好测试远不止“加个 _test.go 文件”这么简单。真正影响项目长期可维护性的,是测试组织方式、执行策略和边界处理——尤其当团队开始写集成测试、基准测试或时间敏感逻辑时,错误模式会集中爆发。
testing.Short() 区分单元与集成测试开发中频繁运行全部测试会拖慢反馈速度,而手动删注释或改文件名又极易出错。Go 官方推荐的解法是统一用 testing.Short() 做守门人。
if testing.Short() { t.Skip("skipping integration test") }
go test -short 时,这些测试自动跳过;CI 或发布前再跑完整版 go test
//go:build integration)更灵活:无需为每个集成测试文件单独打标签,也不用记 --tags=integration 参数t.Skip 判断——如果初始化阶段就连接了 DB,哪怕跳过测试主体,连接动作仍会执行。务必把 testing.Short() 检查放在资源创建之前当你发现 TestValidateEmail 里重复写了五次 t.Run(...) + if result != want,
你就该换写法了。表格驱动不是语法糖,是降低维护成本的核心机制。
cases := []struct{...} 里加一行,不用复制粘贴整个函数体if got != want 容易写反成 if want != got,而表格结构天然约束变量顺序t.Run 内部用 tc(循环变量副本),导致所有子测试都用最后一个 case 的值 —— 必须写成 for _, tc := range cases { t.Run(tc.name, func(t *testing.T) { ... }) }
go test -bench 为什么总测不准?三个硬性条件必须满足基准测试不是“跑一遍看耗时”,它要排除编译器优化、内存抖动和初始化噪声。不满足以下任一条件,数据就不可信。
runtime.KeepAlive / blackhole 变量json.Unmarshal 性能,但先用 bytes.NewReader 构建数据源的过程不能计入耗时 —— 在初始化后立刻调用 b.ResetTimer()
go test -bench 可能只跑几轮就停。应加 -count=5 -benchtime=3s 获取稳定均值,并用 benchstat 工具比对前后差异本地跑一次通过,CI 上偶发失败?大概率是资源没清干净。数据库连接、临时文件、HTTP 服务端口……这些不会自动回收。
defer 配合显式终止:比如用 testcontainers-go 启动 PostgreSQL,必须 defer container.Terminate(ctx),不能只靠 defer server.Close()
test 库 —— 否则并发执行时会相互污染os.Setenv,必须在 defer 中用 os.Unsetenv 恢复,否则影响后续测试TestAdd,而是让第 200 个集成测试在 CI 上稳定通过、让基准测试数据能真实反映一次重构的性能损益。这些细节不体现在文档首页,但每天都在悄悄决定你花在修测试上的时间,到底占开发的 5% 还是 50%。