Are there gotchas using varargs with reference parameters

前端 未结 6 1864
臣服心动
臣服心动 2020-11-29 08:11

I have this piece of code (summarized)...

AnsiString working(AnsiString format,...)
{
    va_list argptr;
    AnsiString buff;

    va_start(argptr, format);         


        
相关标签:
6条回答
  • 2020-11-29 08:38

    According to C++ Coding Standards (Sutter, Alexandrescu):

    varargs should never be used with C++:

    They are not type safe and have UNDEFINED behavior for objects of class type, which is likely causing your problem.

    0 讨论(0)
  • 2020-11-29 08:40

    If you look at what va_start expands out to, you'll see what's happening:

    va_start(argptr, format); 
    

    becomes (roughly)

    argptr = (va_list) (&format+1);
    

    If format is a value-type, it gets placed on the stack right before all the variadic arguments. If format is a reference type, only the address gets placed on the stack. When you take the address of the reference variable, you get the address or the original variable (in this case of a temporary AnsiString created before calling Broken), not the address of the argument.

    If you don't want to pass around full classes, your options are to either pass by pointer, or put in a dummy argument:

    AnsiString working_ptr(const AnsiString *format,...)
    {
        ASSERT(format != NULL);
        va_list argptr;
        AnsiString buff;
    
        va_start(argptr, format);
        buff.vprintf(format->c_str(), argptr);
    
        va_end(argptr);
        return buff;
    }
    
    ...
    
    AnsiString format = "Hello %s";
    s1 = working_ptr(&format, "World");
    

    or

    AnsiString working_dummy(const AnsiString &format, int dummy, ...)
    {
        va_list argptr;
        AnsiString buff;
    
        va_start(argptr, dummy);
        buff.vprintf(format.c_str(), argptr);
    
        va_end(argptr);
        return buff;
    }
    
    ...
    
    s1 = working_dummy("Hello %s", 0, "World");
    
    0 讨论(0)
  • 2020-11-29 08:40

    Here's my easy workaround (compiled with Visual C++ 2010):

    void not_broken(const string& format,...)
    {
      va_list argptr;
      _asm {
        lea eax, [format];
        add eax, 4;
        mov [argptr], eax;
      }
    
      vprintf(format.c_str(), argptr);
    }
    
    0 讨论(0)
  • 2020-11-29 08:43

    Here's what the C++ standard (18.7 - Other runtime support) says about va_start() (emphasis mine) :

    The restrictions that ISO C places on the second parameter to the va_start() macro in header <stdarg.h> are different in this International Standard. The parameter parmN is the identifier of the rightmost parameter in the variable parameter list of the function definition (the one just before the ...). If the parameter parmN is declared with a function, array, or reference type, or with a type that is not compatible with the type that results when passing an argument for which there is no parameter, the behavior is undefined.

    As others have mentioned, using varargs in C++ is dangerous if you use it with non-straight-C items (and possibly even in other ways).

    That said - I still use printf() all the time...

    0 讨论(0)
  • 2020-11-29 08:51

    Side note:

    The behavior for class types as varargs arguments may be undefined, but it's consistent in my experience. The compiler pushes sizeof(class) of the class's memory onto the stack. Ie, in pseudo-code:

    alloca(sizeof(class));
    memcpy(stack, &instance, sizeof(class);
    

    For a really interesting example of this being utilized in a very creative way, notice that you can pass a CString instance in place of a LPCTSTR to a varargs function directly, and it works, and there's no casting involved. I leave it as an exercise to the reader to figure out how they made that work.

    0 讨论(0)
  • 2020-11-29 08:57

    A good analysis why you don't want this is found in N0695

    0 讨论(0)
提交回复
热议问题