
子测试是Go 1.7引入的机制,用于在单个测试函数内组织多个逻辑相关的测试用例,共享setup/teardown,支持独立运行、过滤和并行控制。
testing.T 的子测试(Subtest)子测试是 Go 1.7 引入的机制,用于在单个测试函数内组织多个逻辑相关的测试用例,共享 setup/teardown,同时支持独立运行、过滤和并行控制。它不是嵌套调用 t.Run() 就算——关键在于每个子测试必须有自己的名字、独立生命周期,且父测试函数不能提前返回或 panic,否则后续子测试不会执行。
t.Run() 定义子测试错误写法:把 t.Run() 放在循环外、或漏掉名字参数、或在子测试里直接调用 t.Fatal() 导致整个测试函数退出。正确做法是确保每个子测试有唯一名称,并在闭包中捕获循环变量。
"valid_input"、"empty_string"
for 循环中创建子测试,需将迭代变量显式传入闭包,避免所有子测试共用最后一个值t.Fatal() 只终止当前子测试,不影响其他子测试运行t.Parallel(),但仅当父测试也调用了 t.Parallel() 才真正并发func TestParseURL(t *testing.T) {
tests := []struct {
name string
input string
wantErr bool
}{
{"valid_http", "http://example.com", false},
{"invalid_scheme", "ftp://bad", true},
{"empty", "", t
rue},
}
for _, tt := range tests {
tt := tt // 必须!防止闭包捕获循环变量引用
t.Run(tt.name, func(t *testing.T) {
t.Parallel() // 可选,但需父测试未阻塞
_, err := url.Parse(tt.input)
if (err != nil) != tt.wantErr {
t.Errorf("Parse(%q) error = %v, wantErr %v", tt.input, err, tt.wantErr)
}
})
}
}
testing.T 实例不可复用,每个子测试获得一个全新 *testing.T。这意味着:
t.Run() 闭包内重新初始化t.Cleanup() 注册的清理函数只对当前测试(含子测试)生效;父测试的 Cleanup 不会作用于子测试TestParseURL/valid_http,便于定位go test -run="TestParseURL/valid_http" 可单独运行某个子测试,这对调试非常关键子测试不是银弹。滥用会导致日志冗余、并行度失控、setup 成本重复。尤其注意:
立即学习“go语言免费学习笔记(深入)”;
go test 启动速度,因每个子测试都需注册和调度/ 是合法的,但不要手动拼接层级如 "group/subgroup/case"——Go 会自动解析为嵌套结构,过度嵌套无实际收益最易被忽略的一点:子测试的 t.Log() 输出默认不显示,除非测试失败或加了 -v 参数。调试时别只盯着终端沉默——记得加 go test -v。