从参数包到var args

时间:2016-08-11 12:47:36

标签: c++ c++11

上下文:我处理遗留代码,虽然我想慢慢使这些代码达到c ++ 14标准,但我仍然坚持像printf这样的函数。 因此这种代码(简化):

 #include <iostream>
 #include <string>
 #include <cstdio>

 template <typename... Args>
 const char* Format(const char* strFormat, Args... args)
 {
   static char szBuffer[10000];

   auto len = std::vsnprintf(szBuffer, 10000, strFormat, args...);

   if (len < 0 || 10000 <= len)
   {
     szBuffer[0] = 0;
   }

   return szBuffer;    
 }

 int main()
 {
   auto test = Format("%s %s %d", "test", "test", 42);
 }

这奇怪的不起作用:gcc&amp; VS2013无法从扩展参数包转到va_arg。 (http://cpp.sh/4aue

10:66: error: cannot convert 'const char*' to '__va_list_tag*' for argument '4' to 'int vsnprintf(char*, size_t, const char*, __va_list_tag*)'

为什么会出现这种情况? 同时奇怪的是中间的var_arg函数编译&amp;正常工作(http://cpp.sh/2ggms)。

谢谢

1 个答案:

答案 0 :(得分:2)

您需要snprintf,而不是vsnprintf。参数包扩展为以逗号分隔的列表。这些函数的v版本只接受va_list

在查看代码示例时,如果您打算传递transform_to_c,则仍需要std::string之类的函数。

附注:您应该考虑此代码是否需要是线程安全的还是可重入的。使用静态缓冲区意味着它既不是。