Go并发测试关键在于暴露真实问题:用-go test -race检测竞态,-v查看详情;手动构造goroutine时用sync.WaitGroup控制生命周期,避免sleep等待,错误通过channel或原子变量收集。

Go 的 testing 包原生支持并发测试,但直接用 go test 跑并发逻辑容易漏掉竞态、死锁或时序问题。关键不是“能不能测”,而是“怎么设计才能暴露真实问题”。
-race 检测数据竞争Go 自带的竞态检测器是并发测试的第一道防线,它能发现共享变量未加锁、map 并发读写等典型问题。
go test -race -v ./...(加上 -v 看详细输出)-race 后程序变慢、内存占用高,仅用于测试,不可用于生产sync/atomic 正确操作整数,它不会误报;而对自定义结构体字段做原子操作,需确保用 atomic.Value 或显式同步sync.WaitGroup 控制生命周期单元测试里不能依赖“sleep 等待”,必须精确控制并发 goroutine 的启停和完成信号。
sync.WaitGroup 记录启动的 goroutine 数量,每个 goroutine 结束前调用 wg.Done()
wg.Wait() 阻塞等待全部完成,避免测试提前退出t.Fatal —— 它只影响当前 goroutine,主测试可能已结束。改用 t.Errorf + 共享错误 channel 或原子变量收集问题var wg sync.WaitGroup<br>errCh := make(chan error, 10)<br>for i := 0; i < 10; i++ {<br> wg.Add(1)<br> go func(id int) {<br> defer wg.Done()<br> if err := doWork(id); err != nil {<br> errCh <- fmt.Errorf("worker %d failed: %w", id, err)<br> }<br> }(i)<br>}<br>wg.Wait()<br>close(errCh)<br>for err := range errCh {<br> t.Error(err)<br>}testify/assert 或原生 t.Helper() 提升断言可读性并发测试中失败位置难定位,辅助函数能减少样板代码、统一错误上下文。
立即学习“go语言免费学习笔记(深入)”;
t.Errorf("worker-%d: expected %v, got %v", id, want, got)
t.Helper() 标记自定义检查函数,让错误行号指向调用处而非内部实现t.Fatalf —— 它会终止整个测试。优先用 t.Errorf 收集所有失败,最后统一判断是否要失败testify/assert,注意其断言函数(如 assert.Equal)默认不中断执行,适合批量校验context.Context 和 time.AfterFunc
真实并发逻辑常涉及超时控制、主动取消,测试需覆盖这些边界路径。
time.Sleep 等待结果,改用 select + context.WithTimeout 显式设定等待窗口ctx, cancel := context.WithCancel(context.Background()),在适当时机调用 cancel(),验证被调用函数是否及时退出time.AfterFunc 模拟延迟触发事件(如定时重试、心跳超时),比固定 sleep 更可靠基本上就这些。并发测试不复杂但容易忽略时序和同步细节,核心是“可控、可观测、可复现”——用 -race 揭露隐患,用 WaitGroup 和 Context 控制流程,用结构化断言明确失败原因。
以上就是如何使用Golang测试并发代码_Golang Go test并发场景测试技巧的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号