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
.
将参数包扩展为varargs是有效的.
当你想转发时,转发没有坏处.但是,通过const&
也传达一些有用的东西.
传递给的值...
将经历"默认参数促销".见http://en.cppreference.com/w/cpp/language/variadic_arguments
这些都不关心参考文献.
您可以改进代码.您可以Ts...
通过"类型类型"和实际解析格式化字符串以及确认参数的数量(有时是类型)来检查传递给打印例程的有效类型.如果失败,您可以记录错误消息而不是崩溃.