Go 中 table-driven 测试通过结构体切片统一管理测试用例,用 t.Run 实现子测试并推荐具名结构体、语义化字段名、覆盖边界场景,提升可读性、可维护性与调试效率。
在 Go 中,table-driven 测试是组织多个测试用例最清晰、最可维护的方式。它把输入、预期输出和描述封装成结构体切片,用一个循环统一执行断言,避免重复代码,也便于新增或禁用个别用例。
为每个测试用例定义一个匿名或具名结构体,字段通常包括:name(用于日志定位)、input(函数入参)、want(期望结果),以及可选的 errWant(期望错误)。结构体字段名应语义明确,方便阅读。
func() bool),保持数据与行为分离在 TestXxx 函数中,遍历测试表,对每项调用被测函数,并用 t.Run 创建子测试。子测试名建议包含 tt.name,这样失败时能快速定位具体用例。
t.Run(tt.name, func(t *testing.T) { ... }) 是关键,支持并发运行、独立计时、单独跳过require 或 assert(来自 testify)提升可读性;原生可用 if got != tt.want { t.Errorf(...) }
errors.Is 或 strings.Contains),而非直接 == 错误值一张好的测试表应体现多样性:正常值、零值、空值、超限值、错误输入。例如测试字符串截取函数,可包含:
检查索引越界是否返回 error若某些用例需要前置初始化(如创建临时文件、启动 mock server),可在子测试内部按需执行,不影响其他用例。也可借助 t.Cleanup 注册清理逻辑,确保每个子测试收尾干净。
t.Parallel() 加速,但需确认被测逻辑线程安全//nolint:govet 或 t.Skipf 临时跳过不稳定用例,后续再修复不复杂但容易忽略的是:给每个 tt.name 起个有信息量的名字,比如 "empty_string_returns_empty" 而不是 "case1" —— 这会让测试输出自带文档属性,调试时省一半力气。