C++代码覆盖率测试需用gcc的-fprofile-arcs和-ftest-coverage插桩编译,运行生成.gcda数据,再通过lcov过滤整合并生成HTML报告;编译禁用优化,确保.gcno与.gcda同目录且路径无软链接。

在 C++ 项目中做代码覆盖率测试,核心是用 gcc 的 -fprofile-arcs -ftest-coverage 编译插桩,运行程序生成 .gcda 数据,再用 gcov 解析为行级覆盖报告。真正实用的工程化流程,离不开 lcov 整合多文件、过滤头文件/第三方代码,并生成带样式的 HTML 报告。
编译时开启覆盖率插桩
必须用 gcc 或 g++(Clang 也支持类似选项,但 gcov/lcov 生态主要适配 GCC),且不能开启优化(如 -O2)——否则插桩可能被优化掉,导致覆盖率失真。
- 编译命令加两个关键 flag:
-fprofile-arcs -ftest-coverage - 链接时也要加
-fprofile-arcs(否则会报 undefined reference 到__gcov_flush等符号) - 推荐统一用
cmake管理:在CMakeLists.txt中设置set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fprofile-arcs -ftest-coverage"),并确保CMAKE_BUILD_TYPE为Debug或RelWithDebInfo,禁用-O2/-O3
运行程序生成 .gcda 覆盖数据
插桩后的可执行文件每次运行,都会在对应源文件目录下生成 .gcda 文件(Coverage Data)。注意:.gcda 是增量写入,多次运行会累加统计;若想重跑覆盖率,需手动删除旧 .gcda(或用 lcov --zerocounters)。
- 确保程序正常退出(非 kill/segfault),否则部分计数器可能未刷新到磁盘
- 如果程序是 daemon 或 fork 多进程,需在退出前显式调用
__gcov_flush()(需#include),或设置环境变量GCOV_PREFIX避免权限/路径问题 - 单元测试建议用
ctest或脚本批量执行,保证所有测试用例都跑过
用 lcov 提取、过滤并生成 HTML 报告
gcov 本身只对单个源文件输出文本报告,而 lcov 是基于它的封装工具,专为项目级覆盖率设计。
立即学习“C++免费学习笔记(深入)”;
- 先捕获基础覆盖率:
lcov --capture --directory . --output-file coverage_base.info - 过滤掉不需要统计的路径(如第三方库、测试代码、头文件):
lcov --remove coverage_base.info '/usr/*' '*/test/*' '*mock*' '*.h' --output-file coverage_filtered.info - 生成 HTML:
genhtml coverage_filtered.info --output-directory coverage_report - 打开
coverage_report/index.html即可查看函数/行/分支覆盖率,点击文件可看高亮着色的源码
常见问题与避坑提示
实际使用中,几个细节容易卡住:
-
red">.gcno 文件必须和 .gcda 在同一目录,且编译路径不能有软链接跳转 —— 否则
lcov找不到源码,显示 “No data found” - 头文件(
.h)默认不生成.gcda,但若被inline函数或模板实例化影响,也可能有覆盖数据;建议在lcov --remove中明确排除*.h - 使用
cmake + ninja时,ninja install可能覆盖构建目录结构,导致.gcda路径错乱;推荐在 build 目录内直接运行测试 - CI 流程中,可用
lcov --summary coverage_filtered.info提取总覆盖率数值,配合阈值做门禁(如要求行覆盖 ≥ 80%)
基本上就这些。gcov + lcov 不复杂但容易忽略路径和编译一致性,只要插桩、运行、收集三步稳住,覆盖率就能真实反映测试质量。











