0

0

如何为c++项目编写有效的Property-Based Testing? (autocheck/RapidCheck)

尼克

尼克

发布时间:2026-01-10 11:47:02

|

598人浏览过

|

来源于php中文网

原创

靠谱,但需手动定制生成器、严谨定义性质并限制CI参数。RapidCheck成熟可用,Autocheck已停更;须重载生成器避免脏数据,property需规避浮点误差与全局状态,CI中应固定seed、减少次数并禁用shrink。

如何为c++项目编写有效的property-based testing? (autocheck/rapidcheck)

Property-Based Testing 在 C++ 项目里到底靠不靠谱?

靠。但不是“开箱即用”。RapidCheckAutocheck 都是 C++ 的 property-based testing(PBT)库,其中 RapidCheck 更成熟、文档更全、社区更活跃;Autocheck 已基本停止维护(最后提交在 2018 年),不建议新项目选用。

怎么让 RapidCheck 生成符合业务语义的测试数据?

PBT 失败往往不是因为断言错,而是生成的数据太“脏”——比如传入空指针、负数长度、非法 UTF-8 字节序列,导致被测函数提前崩溃,根本没走到你要验证的性质(property)上。

  • rc::gen::arbitrary() 做起点,但**必须重载或组合生成器**:例如对 std::string,默认生成含嵌入 null、控制字符、超长字符串;应改用 rc::gen::string(rc::gen::alpha()) 限定字符集
  • 对自定义类型,必须显式提供 rc::Gen 特化,不能依赖 ADL 或隐式构造——否则会 fallback 到 bit-wise 随机填充,大概率触发 UB
  • 避免在生成器里做复杂逻辑(如“生成一个递增 vector”):应改用 rc::gen::filter() + 简单生成器,否则 shrink 会失效或极慢
auto genValidUrl = rc::gen::filter([](const std::string& s) {
  return !s.empty() && s.find("://") != std::string::npos;
}, rc::gen::string());

为什么 assert(property) 总是失败,但手动测却没问题?

常见原因是 property 描述不严谨,或忽略了浮点、并发、时序等非确定性因素。

  • assert(x + y == y + x)float 不成立——应改用近似相等:std::abs(x + y - (y + x))
  • 若 property 涉及全局状态(如单例、静态变量)、时间(std::time(nullptr))、随机数(std::rand()),每次运行行为不同,shrink 会失败,RapidCheck 可能报 Failed to shrink 或直接跳过
  • 不要写 “排序后等于 sorted(input)” 这类循环依赖 property:应拆成两个独立检查——is_sorted(output)multiset_equal(input, output)

如何把 PBT 集成进 CI 而不拖慢构建?

默认情况下,RapidCheck 每个 property 运行 100 次用例,且默认开启 shrink(可能额外跑数百次)。CI 中应严格限制:

播记
播记

播客shownotes生成器 | 为播客创作者而生

下载

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

  • rc::prop::forAll(...).run({.tests = 50, .seed = 42}) 固定 seed 和次数,避免 flaky
  • 禁用 shrink 在 CI:加 .run({.shrink = false}) ——定位 bug 是开发机的事,CI 只需快速反馈是否“当前代码通过基础性质”
  • 把 PBT 测试单独编译为 test_pbt 可执行文件,CI 中用 timeout 60 ./test_pbt 控制最长耗时

真正难的是设计好 property:它得足够强以捕获 bug,又不能太强而误报。这点没有捷径,只能从最朴素的不变量开始——比如“parse + serialize 应等于原输入”,再逐步叠加约束。

相关专题

更多
string转int
string转int

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

315

2023.08.02

css中float用法
css中float用法

css中float属性允许元素脱离文档流并沿其父元素边缘排列,用于创建并排列、对齐文本图像、浮动菜单边栏和重叠元素。想了解更多float的相关内容,可以阅读本专题下面的文章。

556

2024.04.28

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

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

98

2025.10.23

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

231

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

435

2024.03.01

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

253

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

206

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1463

2023.10.24

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

25

2026.01.09

热门下载

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

精品课程

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

共18课时 | 4.4万人学习

Sass 教程
Sass 教程

共14课时 | 0.7万人学习

Pandas 教程
Pandas 教程

共15课时 | 0.9万人学习

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

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