0

0

如何使用Google Benchmark为c++代码编写微基准测试? (避免测量误差)

尼克

尼克

发布时间:2026-01-12 12:45:11

|

591人浏览过

|

来源于php中文网

原创

Benchmark::DoNotOptimize并非万能,需配合ClobberMemory()防止计算重排或消除,且须确保结果被真正使用;手动计时破坏Google Benchmark统计模型,应使用PauseTiming()而非std::chrono;BENCHMARK_TEMPLATE易致模板爆炸,宜用constexpr if替代多特化。

如何使用google benchmark为c++代码编写微基准测试? (避免测量误差)

为什么 Benchmark::DoNotOptimize 不是万能的

直接把待测函数包裹在 Benchmark::DoNotOptimize 里,常被误认为“防止优化就万事大吉”。实际中,编译器仍可能将整个计算提前到基准循环外(尤其是无副作用的纯函数),或把多次调用合并为一次。关键在于:该函数只阻止值被优化掉,不阻止计算本身被重排或消除。

  • 必须配合 Benchmark::DoNotOptimize + Benchmark::ClobberMemory() 使用,后者强制编译器刷新寄存器和内存别名状态
  • 若函数返回值参与后续逻辑(如累加),需确保结果真正被使用——例如赋给一个 volatile 变量,或传入 Benchmark::DoNotOptimize 后立即读取
  • 对修改全局状态的函数(如填充 vector),还要注意 std::vector::reserve 是否被提前调用,否则内存分配开销会污染测量

如何正确设置 BENCHMARK_MAIN() 和运行参数

默认 BENCHMARK_MAIN() 会启用所有内置计时器,但某些环境(如虚拟机、容器)下 CPU 频率动态调整会导致抖动。必须显式控制基准行为,而非依赖默认。

  • 通过命令行传参比硬编码更灵活:
    ./benchmark --benchmark_repetitions=5 --benchmark_report_aggregates_only=true --benchmark_format=json
  • --benchmark_min_time=1 比默认的 0.5 秒更稳妥,避免短函数因计时精度不足产生噪声
  • 禁用 CPU 频率缩放(Linux):echo performance | sudo tee /sys/devices/system/cpu/cpu*/cpufreq/scaling_governor,否则 ns_per_iteration 波动可能超 20%

避免 std::chrono 手动计时与 Google Benchmark 混用

有人试图在 Benchmark::State 循环内用 std::chrono::high_resolution_clock 手动测单次耗时,再除以迭代次数——这完全破坏了 Benchmark 的统计模型。Google Benchmark 不是简单取平均,它会剔除异常值、拟合置信区间、检测抖动,并自动调整迭代次数使误差低于阈值。

Google AI Studio
Google AI Studio

Google 推出的基于浏览器的集成开发环境

下载
  • 手动计时绕过了 Benchmark::State::PauseTiming()ResumeTiming() 的精确控制点,预热、缓存效应、TLB miss 等都无法被建模
  • 若需测量某段子逻辑(比如排除 I/O),应拆分为独立 benchmark 函数,用 state.PauseTiming() 包裹非目标代码,而非引入外部时钟
  • 验证是否生效:运行后检查输出中的 iterations 字段——正常情况应远大于 1(如 10000+),若恒为 1,说明计时逻辑已被破坏

为什么 BENCHMARK_TEMPLATE 容易引发实例化爆炸

对模板函数写 BENCHMARK_TEMPLATE(my_func, int) 看似简洁,但每个特化都会生成独立符号和计时逻辑。当类型参数多于 1 个、或存在 std::vector 这类嵌套模板时,编译时间和二进制体积会指数增长,且不同特化间无法共享预热状态。

立即学习C++免费学习笔记(深入)”;

  • 优先用 auto + constexpr if 在单个 benchmark 函数内分路径,而非展开多个特化
  • 若必须测多类型,用宏生成有限组合(如 int/double/std::string),避免泛型参数如 typename T 直接暴露给 BENCHMARK_TEMPLATE
  • 检查编译日志中是否有大量 instantiate 提示——这是模板膨胀的明确信号
真实微基准最难的不是写代码,而是让编译器和 CPU “按你说的做”,而不是“按它想的做”。哪怕一行 Benchmark::ClobberMemory() 漏掉,或一个 --benchmark_min_time 设太小,测出来的数字就只是运行时偶然现象。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

731

2023.08.22

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

534

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

51

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

194

2025.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

51

2025.08.29

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

98

2025.10.23

Java 项目构建与依赖管理(Maven / Gradle)
Java 项目构建与依赖管理(Maven / Gradle)

本专题系统讲解 Java 项目构建与依赖管理的完整体系,重点覆盖 Maven 与 Gradle 的核心概念、项目生命周期、依赖冲突解决、多模块项目管理、构建加速与版本发布规范。通过真实项目结构示例,帮助学习者掌握 从零搭建、维护到发布 Java 工程的标准化流程,提升在实际团队开发中的工程能力与协作效率。

3

2026.01.12

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 7万人学习

Git 教程
Git 教程

共21课时 | 2.6万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号