go test 不能替代调试,二者目标、手段、阶段均不同:测试是“找错”,调试是“定位并修错”;单元测试覆盖有限,难发现竞态、泄漏、第三方失败等问题,而调试可深入运行时状态精准修复。
go test 不能替代调试,二者目标不同、手段不同、作用阶段也不同。测试是“找错”,调试是“定位并修错”——就像体检报告和手术刀的关系:前者告诉你哪里可能有问题,后者才能切开看清楚、动手修复。
go test 发现不了所有问题?单元测试只覆盖你显式编写的用例路径,对未声明的边界、并发竞态、资源泄漏、第三方交互失败等几乎无感。
t.Errorf 只在断言失败时触发,而程序“没崩溃但逻辑歪了”(比如返回了错误的缓存值)常被忽略BenchmarkXxx 关注耗时,但不校验中间状态;go test -bench=. 跑出 0.34 ns/op 很漂亮,可如果函数悄悄改了全局变量,它完全不报警当你看到测试通过但程序在线上卡死、goroutine 泄漏、HTTP 返回 500 却没 panic、或 go test -race

dlv debug 启动,设断点在疑似位置:dlv debug --headless --listen=:2345 --api-version=2
http.HandlerFunc,观察 req.Context().Value() 是否被意外覆盖runtime.Stack() 或 debug.ReadGCStats() 在关键节点打日志,比 fmt.Println 更轻量、更可控testing.T 和 dlv 怎么配合才不重复劳动?测试不是调试的替代品,而是调试的“路标”。写测试时就该埋好调试线索。
TestXxx 里加 t.Log("state:", obj.Status, "cache hit:", hit),运行时用 go test -v 看上下文,比盲猜快得多TestXxxWithTimeout,并在函数内用 if testing.Verbose() { log.Printf(...) 控制日志粒度os.Exit(1) 或 panic——这会让 dlv 失去控制权;改用 t.Fatal,它能被测试框架捕获并继续执行后续清理很多人以为 “go test 覆盖率 95% 就很稳”,但以下三类问题几乎从不进覆盖率统计,却高频导致线上事故:
defer wg.Done(),或 select 漏写了 default 导致永久阻塞dlv 里看到的时间戳永远不对dlv attach 到正在跑的进程里看 goroutine stack;测试写得再密,也只是你认为的路径——而程序实际走的,永远是它自己选的那条。