std::string and stdarg.h
std::string and stdarg.h
我编写了一个函数,它试图通过返回std::字符串而不是写入用户提供的char*
来进行自动分配sprintf。(请不要回答推荐iostreams或Boost.Format或朋友的问题——我知道它们存在,我确实在其他情况下使用它们,但在这种特殊情况下有要求。)
std::string FormatString(const std::string& format, va_list argList)
{
char smallBuffer[500], *text = smallBuffer;
int length = _countof(smallBuffer);
// MSVC is not C99 conformant, so its vsnprintf returns -1
// on insufficient buffer space
int outputSize = _vsnprintf(text, length, format.c_str(), argList);
while (outputSize < 0 && errno == ERANGE && length > 0)
{
length <<= 1;
if (text != smallBuffer) { delete[] text; }
text = new char[length];
outputSize = _vsnprintf(text, length, format.c_str(), argList);
}
if (outputSize < 0)
{
throw std::runtime_error("Failed to format string.");
}
std::string ret(text);
if (text != smallBuffer)
{
delete[] text;
}
return ret;
}
std::string FormatString(const std::string& format, ...)
{
va_list argList;
va_start(argList, format);
std::string result;
try
{
result = FormatString(format, argList);
}
catch(...)
{
va_end(argList);
throw;
}
va_end(argList);
return result;
}
int _tmain(int argc, _TCHAR* argv[])
{
int foo = 1234;
std::string bar = "BlaBla";
std::cout << FormatString("%i (%s)", foo, bar.c_str()) << std::endl;
return 0;
}
(是的,我看到了将C格式的字符串管道连接到C++iostream的讽刺。这只是测试代码。)
不幸的是,使用VS2008,它在printf内部深处崩溃,显然是因为它从va_list
中读取了错误的参数(根据调试器的说法,在va_start
之后,它指向的是紧挨着"真实"第一个参数之前的四字节空序列)。
特别值得注意的是,如果在变差函数中,我将const std::string& format
更改为仅std::string format
(即按值传递),则它可以正常工作;当然,如果我将其更改为const char *
,它也会这样做。
这是某种编译器错误,还是使用带有引用参数的va_list不合法?
如果你想传递引用,我认为你运气不好。以下是C++2011标准在18.10[support.runtime]第3段中对这个主题的描述:
ISO C对标头中va_start()宏的第二个参数的限制在本国际标准中有所不同。参数parmN是函数定义的变量参数列表中最右边参数的标识符(刚好在…之前的那个)。230如果参数parmN是用函数、数组或引用类型声明的,或者是用与传递没有参数的参数时所产生的类型不兼容的类型来声明的,则行为是未定义的。
相关文章:
- cppcheck在const std::string[]上引发警告
- 将std::string传递给WriteConsole API
- 为std::string的某个索引赋值
- std中有类似find_last_of的函数,而string中没有
- 使用 std::string () const 函数启动线程或未来
- 使用char类型将decimal转换为string,将string转换为decimal
- 迭代和比较映射<字符串、矢量<string>> c++ 中的值
- 当我们进行一些操作时,应该使用什么'std::string'或'std::stringstream'?
- 将向量解析<string>为字符串
- 'string.assign(string.data(), 5)' 是明确定义的还是 UB?
- 如何更改大小(std::string)
- "string.h"在构建适用于iOS的qt应用程序中找不到消息
- C++:如何将 unix 时间的字符串转换为 *tm?(使用时间错误:"cannot convert 'String' to 'tm*' ")
- std::string 的对象真的可以移动吗?
- 与'operator='不匹配(操作数类型'String'且"void")
- SegFault 同时使用 std::string::operator+= 和函数作为参数
- 无法从 std::string 中提取C++ Unicode 符号
- std::string 构造函数如何处理固定大小的 char[]?
- <string> 使用 for 循环写入向量
- std::string and stdarg.h