c++性能测试工具:google benchmark入门(二)

本文索引

向测试用例传递参数

之前我们的测试用例都只接受一个 benchmark::State& 类型的参数,如果我们需要给测试用例传递额外的参数呢?

举个例子,假如我们需要实现一个队列,现在有ring buffer和linked list两种实现可选,现在我们要测试两种方案在不同情况下的性能表现:

// 必要的数据结构#include"ring.h"#include"linked_ring.h"// ring buffer的测试staticvoidbench_array_ring_insert_int_10(benchmark::State& state){autoring = ArrayRing(10);for(auto_: state) {for(inti =1; i <=10; ++i) {            ring.insert(i);        }        state.PauseTiming();// 暂停计时ring.clear();        state.ResumeTiming();// 恢复计时}}BENCHMARK(bench_array_ring_insert_int_10);// linked list的测试staticvoidbench_linked_queue_insert_int_10(benchmark::State &state){autoring = LinkedRing{};for(auto_:state) {for(inti =0; i <10; ++i) {            ring.insert(i);        }        state.PauseTiming();        ring.clear();        state.ResumeTiming();    }}BENCHMARK(bench_linked_queue_insert_int_10);// 还有针对删除的测试,以及针对string的测试,都是高度重复的代码,这里不再罗列

很显然,上面的测试除了被测试类型和插入的数据量之外没有任何区别,如果可以通过传入参数进行控制的话就可以少写大量重复的代码。

编写重复的代码是浪费时间,而且往往意味着你在做一件蠢事,google的工程师们当然早就注意到了这一点。虽然测试用例只能接受一个 benchmark::State& 类型的参数,但我们可以将参数传递给state对象,然后在测试用例中获取:

staticvoidbench_array_ring_insert_int(benchmark::State& state){autolength = state.range(0);autoring = ArrayRing(length);for(auto_: state) {for(inti =1; i <= length; ++i) {            ring.insert(i);        }        state.PauseTiming();        ring.clear();        state.ResumeTiming();    }}BENCHMARK(bench_array_ring_insert_int)->Arg(10);

上面的例子展示了如何传递和获取参数:

传递参数使用 BENCHMARK 宏生成的对象的 Arg 方法

传递进来的参数会被放入state对象内部存储,通过 range 方法获取,调用时的参数0是传入参数的需要,对应第一个参数

Arg 方法一次只能传递一个参数,那如果一次想要传递多个参数呢?也很简单:

staticvoidbench_array_ring_insert_int(benchmark::State& state){autoring = ArrayRing(state.range(0));for(auto_: state) {for(inti =1; i <= state.range(1); ++i) {            ring.insert(i);        }        state.PauseTiming();        ring.clear();        state.ResumeTiming();    }}BENCHMARK(bench_array_ring_insert_int)->Args({10,10});

上面的例子没什么实际意义,只是为了展示如何传递多个参数, Args方法接受一个vector对象,所以我们可以使用c++11提供的大括号初始化器简化代码,获取参数依然通过 state.range 方法,1对应传递进来的第二个参数。

有一点值得注意,参数传递只能接受整数,如果你希望使用其他类型的附加参数,就需要另外想些办法了。

简化多个类似测试用例的生成

向测试用例传递参数的最终目的是为了在不编写重复代码的情况下生成多个测试用例,在知道了如何传递参数后你可能会这么写:

staticvoidbench_array_ring_insert_int(benchmark::State& state){autolength = state.range(0);autoring = ArrayRing(length);for(auto_: state) {for(inti =1; i <= length; ++i) {            ring.insert(i);        }        state.PauseTiming();        ring.clear();        state.ResumeTiming();    }}// 下面我们生成测试插入10,100,1000次的测试用例BENCHMARK(bench_array_ring_insert_int)->Arg(10);BENCHMARK(bench_array_ring_insert_int)->Arg(100);BENCHMARK(bench_array_ring_insert_int)->Arg(1000);

这里我们生成了三个实例,会产生下面的结果:

看起来工作良好,是吗?

没错,结果是正确的,但是记得我们前面说过的吗—— 不要编写重复的代码 !是的,上面我们手动编写了用例的生成,出现了可以避免的重复。

幸好 Arg 和 Args 会将我们的测试用例使用的参数进行注册以便产生 用例名/参数 的新测试用例,并且返回一个指向 BENCHMARK 宏生成对象的指针,换句话说,如果我们想要生成仅仅是参数不同的多个测试的话,只需要链式调用 Arg 和 Args 即可:

BENCHMARK(bench_array_ring_insert_int)->Arg(10)->Arg(100)->Arg(1000);

结果和上面一样。

但这还不是最优解,我们仍然重复调用了Arg方法,如果我们需要更多用例时就不得不又要做重复劳动了。

对此google benchmark也有解决办法:我们可以使用 Range 方法来自动生成一定范围内的参数。

先看看Range的原型:

BENCHMAEK(func)->Range(int64_tstart, int64_t limit);

start表示参数范围起始的值,limit表示范围结束的值,Range所作用于的是一个_闭区间_。

但是如果我们这样改写代码,是会得到一个错误的测试结果:

BENCHMARK(bench_array_ring_insert_int)->Range(10,1000);

为什么会这样呢?那是因为Range默认除了start和limit,中间的其余参数都会是某一个基底(base)的幂,基地默认为8,所以我们会看到64和512,它们分别是8的平方和立方。

想要改变这一行为也很简单,只要重新设置基底即可,通过使用 RangeMultiplier 方法:

BENCHMARK(bench_array_ring_insert_int)->RangeMultiplier(10)->Range(10,1000);

现在结果恢复如初了。

使用Ranges可以处理多个参数的情况:

BENCHMARK(func)->RangeMultiplier(10)->Ranges({{10,1000}, {128,256}});

第一个范围指定了测试用例的第一个传入参数的范围,而第二个范围指定了第二个传入参数可能的值(注意这里不是范围了)。

与下面的代码等价:

BENCHMARK(func)->Args({10,128})->Args({100,128})->Args({1000,128})->Args({10,256})->Args({100,256})->Args({1000,256})

实际上就是用生成的第一个参数的范围于后面指定内容的参数做了一个笛卡尔积。

使用参数生成器

如果我想定制没有规律的更复杂的参数呢?这时就需要实现自定义的参数生成器了。

一个参数生成器的签名如下:

void CustomArguments(benchmark::internal::Benchmark*b);

我们在生成器中计算处参数,然后调用 benchmark::internal::Benchmark 对象的Arg或Args方法像上两节那样传入参数即可。

随后我们使用 Apply 方法把生成器应用到测试用例上:

BENCHMARK(func)->Apply(CustomArguments);

其实这一过程的原理并不复杂,我做个简单的解释:

BENCHMARK 宏产生的就是一个 benchmark::internal::Benchmark 对象然后返回了它的指针

向 benchmark::internal::Benchmark 对象传递参数需要使用Arg和Args等方法

Apply 方法会将参数中的函数应用在自身

我们在生成器里使用 benchmark::internal::Benchmark 对象的指针b的Args等方法传递参数,这时的b其实指向我们的测试用例

到此为止生成器是如何工作的已经一目了然了,当然从上面得出的结论,我们还可以让Apply做更多的事情。

下面看下Apply的具体使用:

// 这次我们生成100,200,...,1000的测试用例,用range是无法生成这些参数的staticvoidcustom_args(benchmark::internal::Benchmark* b){for(int i =100; i <=1000; i +=100) {b->Arg(i);    }}BENCHMARK(bench_array_ring_insert_int)->RangeMultiplier(10)->Apply(custom_args);

自定义参数的测试结果:

至此向测试用例传递参数的方法就全部介绍完了。

下一篇中我会介绍如何将测试用例写成模板,传递参数只能解决一部分重复代码,对于拥有类似方法的不同待测试类型的测试用例,使用模板将会大大减少我们不必要的工作。

看我主页简介免费C++学习资源,视频教程、职业规划、面试详解、学习路线、开发工具

每晚8点直播讲解C++编程技术。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351

推荐阅读更多精彩内容