我坚持的代码是伪格式化函数,它接受一个格式字符串和零个或多个参数(const char * format,…)作为输入.然后它将处理一些占用一些参数的占位符,并将其余的传递给vsprintf以及动态生成的新va_list.
char *new_args = (char *) malloc(sum); char *n = new_args; for(int i = 0; i < nArgs; i++) { int j = order[i]; int len = _getlen(types[j]); memcpy(n,args + cumulOffsets[j],len); n += len; } vsprintf(buffer,sFormat.c_str(),new_args);
在我的辩护中,我没有,也绝不会写这段代码.事实上,我认为这是我一生中见过的最讨厌的事情之一.
但是,这个功能非常复杂,非常陈旧,非常重要.它也没有被修改多年(好吧,除了现在),所以虽然我想从头开始重写它,但我无法证明它需要的时间加上它会引入的错误.
所以,我需要一种方法在GCC上做同样的事情..但是有一个va_list不是char *所以我得到:
error: ISO C++ forbids casting to an array type '__va_list_tag [1]'
解决方法
我相信vsnprintf()使用当前的va_list对象(如果你可以调用它).所以你可以自由地使用va_start(),通过va_arg()使用你想要的参数,然后通过va_list将剩下的参数传递给vsnprintf(),然后调用va_end().
我错过了什么吗?为什么深拷贝?
如果你确实需要深层复制,为什么不刷新va_start(),通过va_arg()删除你想要的参数,然后将生成的va_list对象传递给vsnprintf().
(每次调用va_arg都会修改va_list对象,以便下一个调用返回下一个参数.)
或者,您可以使用va_copy(). (尽管一定要用相应的va_end()跟随它.)
附录:另请注意,这些va_宏基于C89& C99标准. GNU g将支持它们.微软有点受限.
跟进TonyK的评论:
如果您从va_list中删除前N个项目,我上面所说的就可以了.如果你从中间拉出物品,那就更难了.
没有可移植的方法来构造va_list.
但是,您可以拆分格式字符串,使用它来确定对象类型(double,float,int等),并使用它自己的格式字符串(原始格式字符串的子部分)单独打印每个字符串.多个snprintf()调用将导致一些开销.但如果不经常调用这个例程,它应该是可行的.
您还可以使用适当制作的va_list打印出原始格式字符串的子部分.换句话说,第一个vsnprintf()调用打印元素1..3,第二个元素5..7,第三个10..13等等(因为vsnprintf()会忽略va_list上的额外元素超出它的需要你只需要一系列相应的格式字符串片段,并根据每个vsnprintf()调用的需要用va_arg()弹出va_list中的项目.)