将可变参数模板参数传递给可变参数函数

Tri*_*dle 9 c++ variadic-functions variadic-templates perfect-forwarding c++11

我们正在使用提供printf()样式日志功能的第三方C库,

void log(const char *format, ...);
Run Code Online (Sandbox Code Playgroud)

由于不值得进入的原因,我们需要限制消息记录的速率,这有点像

void rate_limited_log(const char* format, ...)
{
    if (<not too fast>) {
         log(format, ...);
    }
}
Run Code Online (Sandbox Code Playgroud)

幸运的是,C库的作者知道他们在做什么,并提供了

void logv(const char* format, va_list ap);
Run Code Online (Sandbox Code Playgroud)

所以写上面的函数是一件相对简单的事情.不幸的是,可变功能在内联方面效果不佳,所以我想出了第二个解决方案:

template <typename... T>
void rate_limited_log(const char* format, T&&... args)
{
    if (<not too fast>) {
        log(format, std::forward<T>(args)...);
    }
}
Run Code Online (Sandbox Code Playgroud)

这完美地工作,并按照我们的意愿内联速率限制条件.但我有几个问题:

  • 是否将参数包扩展为C风格的可变参数函数调用,这样在C++ 11中可以做一个合法的,定义明确的事情,或者让我们幸运的是它有效吗?

  • &&std::forward实际需要在这里,给我们调用C函数?如果我使用const T&,或者甚至仅仅T通过价值,有或没有,它似乎也可行std::forward.

Yak*_*ont 8

将参数包扩展为varargs是有效的.

当你想转发时,转发没有坏处.但是,通过const&也传达一些有用的东西.

传递给的值...将经历"默认参数促销".见http://en.cppreference.com/w/cpp/language/variadic_arguments

这些都不关心参考文献.

您可以改进代码.您可以Ts...通过"类型类型"和实际解析格式化字符串以及确认参数的数量(有时是类型)来检查传递给打印例程的有效类型.如果失败,您可以记录错误消息而不是崩溃.