全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

如何在Golang中实现自定义Benchmark_Golang testing.B自定义性能测量示例

自定义 Benchmark 函数必须命名为 BenchmarkXxx,接收 *testing.B 参数,通过 go test -bench=. 运行;需在 b.N 循环外初始化并调用 b.ResetTimer(),避免测量初始化开销。

如何在 Go test 文件中定义并运行自定义 Benchmark 函数

Go 的 testing.B 是唯一合法的 benchmark 入口,所有自定义性能测量必须以 BenchmarkXxx 形式定义,且函数签名固定为 func(*testing.B)。它不是普通函数调用,不能手动传参或直接执行——必须通过 go test -bench=. 触发。

常见错误是把 benchmark 写成普通函数、漏掉 b.ResetTimer()、或在 b.N 循环外做初始化却未排除耗时,导致结果失真。

  • 函数名必须以 Benchmark 开头,首字母大写
  • 必须接收单个 *testing.B 参数
  • 循环体必须用 for i := 0; i ,不可硬编码次数
  • 预热或初始化逻辑应放在 for 循环之前,并调用 b.ResetTimer() 重置计时起点

为什么必须调用 b.ResetTimer()b.StopTimer()

Go 默认从函数入口开始计时,但初始化(如构造 map、读配置、建连接)不属于被测逻辑。若不干预,这些开销会污染 b.N 循环的平均耗时。

b.ResetTimer() 重置计时器,把后续 b.N 循环作为唯一测量区间;b.StopTimer() 暂停计时,适合在循环中穿插非关键操作(如日志打印、临时缓存清理),避免干扰核心路径。

  • 初始化后、循环前调用 b.ResetTimer() 是最常见且必要的操作
  • 若循环体内需执行不可省略但非目标逻辑(如 sync.Pool Get/Pool),可用 b.StopTimer() + b.StartTimer() 包裹
  • 忘记 ResetTimer() 会导致 benchmark 报出极低 QPS、极高 ns/op,数值完全不可比

如何测量不同输入规模下的性能变化(如 slice 长度递增)

标准 testing.B 不支持参数化 benchmark,但可通过闭包或子测试方式模拟多组输入。推荐在单个 Benchmark 函数内遍历不同规模,对每组调用 b.Run() 创建子 benchmark,便于横向对比。

注意:每个 b.Run() 子项独立计时,且会继承父项的 -benchmem 等标志,适合观察增长趋势。

func BenchmarkCopySlice(b *testing.B) {
	sizes := []int{100, 1000, 10000}
	for _, n := range sizes {
		b.Run(fmt.Sprintf("Len%d", n), func(b *testing.B) {
			data := make([]byte, n)
			b.ResetTimer()
			for i := 0; i < b.N; i++ {
				_ = append([]byte(nil), data...)
			}
		})
	}
}

如何安全地在 benchmark 中使用并发(b.RunParallel

b.RunParallel 用于模拟多 goroutine 并发调用同一逻辑,适用于测试锁竞争、channel 吞吐、sync.Pool 等场景。但它不保证总执行次数为 b.N,而是将 b.N 拆分给多个 goroutine 并行执行——实际调用次数 ≥ b.N,且无法控制 goroutine 数量(由 runtime 自动调度)。

关键约束:被测函数不能依赖共享可变状态,否则结果不可复现;所有初始化必须在 b.RunParallel 外完成。

  • 传入的 func(*testing.PB) 中,pb.Next() 控制是否继续,必须用 for pb.Next() { ... } 循环包裹被测逻辑
  • 不要在 pb.Next() 循环内做初始化(如 new struct、open file),这会放大资源开销
  • 并发 benchmark 的内存分配统计(-benchmem)反映的是单次调用均值,不是总和
func BenchmarkConcurrentMapSet(b *testing.B) {
	m := make(map[int]int)
	var mu sync.RWMutex
	b.ResetTimer()
	b.RunParallel(func(pb *testing.PB) {
		for pb.Next() {
			mu.Lock()
			m[1] = 1
			mu.Unlock()
		}
	})
}
Go benchmark 的真实难点不在写法,而在于区分“测什么”和“怎么排除干扰”。比如 time.Now() 调用本身有开销,fmt.Sprintf 在循环里会掩盖核心逻辑性能,甚至 GC 周期都可能让结果抖动——这些都需要靠多次运行、关闭 GC(debug.SetGCPercent(-1))、或结合 pprof 进一步定位。


# go  # golang  # 编码  # app  # 为什么  # for  # 循环  # 继承  # Struct  # 闭包  # map  # 并发  # channel  # 自定义  # 的是  # 放在  # 多个  # 遍历  # 适用于  # 计时器  # 能让  # 不支持  # 极高 


相关文章: 制作旅游网站html,怎样注册旅游网站?  高性能网站服务器配置指南:安全稳定与高效建站核心方案  建站之星3.0如何解决常见操作问题?  七夕网站制作视频,七夕大促活动怎么报名?  如何破解联通资金短缺导致的基站建设难题?  如何用搬瓦工VPS快速搭建个人网站?  保定网站制作方案定制,保定招聘的渠道有哪些?找工作的人一般都去哪里看招聘信息?  购物网站制作公司有哪些,哪个购物网站比较好?  公司网站设计制作厂家,怎么创建自己的一个网站?  如何通过西部数码建站助手快速创建专业网站?  相亲简历制作网站推荐大全,新相亲大会主持人小萍萍资料?  高防服务器:AI智能防御DDoS攻击与数据安全保障  建站主机是什么?如何选择适合的建站主机?  广东专业制作网站有哪些,广东省能源集团有限公司官网?  如何配置WinSCP新建站点的密钥验证步骤?  建站之星北京办公室:智能建站系统与小程序生成方案解析  建站之星展会模板:智能建站与自助搭建高效解决方案  如何在Ubuntu系统下快速搭建WordPress个人网站?  极客网站有哪些,DoNews、36氪、爱范儿、虎嗅、雷锋网、极客公园这些互联网媒体网站有什么差异?  车管所网站制作流程,交警当场开简易程序处罚决定书,在交警网站查询不到怎么办?  武汉网站如何制作,黄黄高铁武穴北站途经哪些村庄?  如何在阿里云通过域名搭建网站?  如何在橙子建站上传落地页?操作指南详解  云南网站制作公司有哪些,云南最好的招聘网站是哪个?  大连网站制作费用,大连新青年网站,五年四班里的视频怎样下载啊?  C++中引用和指针有什么区别?(代码说明)  简易网站制作视频教程,使用记事本编写一个简单的网页html文件?  建站之星后台密码如何安全设置与找回?  大学网站设计制作软件有哪些,如何将网站制作成自己app?  济南网站建设制作公司,室内设计网站一般都有哪些功能?  制作假网页,招聘网的薪资待遇,会有靠谱的吗?一面试又各种折扣?  网站制作大概多少钱一个,做一个平台网站大概多少钱?  制作网站的基本流程,设计网站的软件是什么?  上海制作企业网站有哪些,上海有哪些网站可以让企业免费发布招聘信息?  c++如何打印函数堆栈信息_c++ backtrace函数与符号名解析【方法】  制作公司内部网站有哪些,内网如何建网站?  如何自定义建站之星模板颜色并下载新样式?  北京网站制作费用多少,建立一个公司网站的费用.有哪些部分,分别要多少钱?  制作营销网站公司,淘特是干什么用的?  淘宝制作网站有哪些,淘宝网官网主页?  如何通过虚拟机搭建网站?详细步骤解析  如何在景安服务器上快速搭建个人网站?  电影网站制作价格表,那些提供免费电影的网站,他们是怎么盈利的?  如何在企业微信快速生成手机电脑官网?  宝塔面板如何快速创建新站点?  建站之星客服服务时间及联系方式如何?  电视网站制作tvbox接口,云海电视怎样自定义添加电视源?  哪家制作企业网站好,开办像阿里巴巴那样的网络公司和网站要怎么做?  如何确保西部建站助手FTP传输的安全性?  如何快速搭建高效WAP手机网站? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。