std :: string和stdarg.h

时间:2012-02-02 23:51:30

标签: c++ printf stdstring variadic-functions

我编写了一个函数,它试图通过返回一个std :: string而不是写入用户提供的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是不合法的?

1 个答案:

答案 0 :(得分:3)

如果您想传递参考,我认为您运气不好。以下是C ++ 2011标准在18.10 [support.runtime]第3段中对该主题的评论:

  

ISO C对标头中va_start()宏的第二个参数的限制在本国际标准中是不同的。参数parmN是函数定义的变量参数列表中最右边的参数的标识符(恰好在......之前).230如果使用函数,数组或引用类型声明参数parmN,或者使用与传递没有参数的参数时产生的类型不兼容的类型,行为未定义。