0

因为除了 github 自述文件之外我找不到真正好的资源或文档,我会在这里询问社区。

假设我有一个名为 的接口Base和多个派生类型。
我想Foo用相同的函数和传递给的相同参数对所有派生类型的虚函数进行基准测试Foo
对于每个派生类型,我想Foo在单个基准测试中使用从std::vector<std::size_t>.

我的一些派生类型的构造函数也需要额外的参数,我应该如何处理它?

在我看来,最简单的做法是使用模板化基准测试并传递给:

const std::vector<std::size_t> v {1, 2, 4, 8, 16};
BENCHMARK_TEMPLATE(func, Derived1, v);
BENCHMARK_TEMPLATE(func, Derived2, v);
BENCHMARK_TEMPLATE(func, Derived3, v);
...

我觉得这不是正确的方法,因为对于每个具有不同构造函数签名的派生类型,我需要具有与前一个几乎相同的不同基准函数。

对此进行基准测试的正确方法是什么?

**更新*:我解决这个问题的方法如下:

std::unique_ptr<Base> GetObj(const std::size_t index)
{
    switch(index)
    {
    case 0:
        return std::make_unique<Derived1>(...);
    case 1:
        return std::make_unique<Derived2>(...);
    ...
    }
}

void func(benchmark::State& state)
{
    auto r = state.range();
    auto object = GetObj(r);
    for (auto _ : state)
    {
        ...
    }
}
BENCHMARK(func)->DenseRange(0,5);

但我不喜欢的是基准测试的名称是模式func/0-5,我希望它是func/Derived1func/Derived2等等..有什么想法可以实现吗?

4

1 回答 1

1

你的方法非常正确。如果您想避免使用多个基准测试函数,请将基准测试的核心移动到一个单独的函数中,并从每个类型的每个函数中调用它。就像是:

static void Benchmark(Base* b, benchmark::State& st) {
  for (auto _ : state) {
    b->DoTheThing()
  }
}

void BM_Foo(benchmark::State& st) {
  Foo f;
  Benchmark(&f, st);
}

BENCHMARK(BM_Foo);

如果您想更复杂,您可以使用该RegisterBenchmark功能更灵活地注册事物。就像是:

auto BM_base = [](benchmark::State& st, auto inputs, Base* b) { /* ... */ };

int main(int argc, char**argv) {
  const std::vector<std::size_t> v = {1, 2, 4, 8, 16};
  benchmark::RegisterBenchmark("foo", BM_base, v, new Foo());
  benchmark::Initialize(&argc, argv);
  return benchmark::RunSpecifiedBenchmarks();
}

Arg顺便说一句,在注册基准时应该将 v 设置为s 吗?

于 2020-06-01T14:44:40.657 回答