微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

c – 如何在GCC上创建va_list?

我正在尝试转换一些代码,以便它也在 gcc上编译(现在,它仅在MSVC上编译).

我坚持的代码是伪格式化函数,它接受一个格式字符串和零个或多个参数(const char * format,…)作为输入.然后它将处理一些占用一些参数的占位符,并将其余的传递给vsprintf以及动态生成的新va_list.

这是生成新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]'

解决方法

我有点迷茫.为什么需要一个新的动态生成的va_list?为什么不重用旧的呢?

我相信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中的项目.)

原文地址:https://www.jb51.cc/c/119331.html

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐