关于c++可变参数的理解

来源:互联网 发布:淘宝销量代码 编辑:程序博客网 时间:2024/05/22 09:42

  最近在看cocos2dx的源码,发现有很多地方用到了可变参数,就是三个点,然后查了查资料,把自己的思路整理一下,希望能对看到的人有帮助。

  首先三个点的值传到了宏__VA_ARGS____VA_ARGS__只能在宏中替换可变参数,,下面举个例子(该例子来源于http://http://my.csdn.net/MoreWindows具体是那篇我忘了)

#include <iostream>#include <stdarg.h>#define WriteLine(...) printf(__VA_ARGS__);int main(){int i = WriteLine("Hello World");printf("\n");WriteLine("%d", i);getchar();//这里用这个是为了让控制台停下来让大家看到结果return 0;}

  然后在http://en.cppreference.com/w/cpp/utility/variadic这上面看了可变参数用的几个低级的函数va_start、va_arg、va_copy(c++11)、va_end和va_list。在vs2013中va_list在vadefs.h就是typedef char *  va_list;这样定义的,下面列出函数的实现

#define _crt_va_start(ap,v)  ( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) )#define _crt_va_arg(ap,t)    ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )#define _crt_va_end(ap)      ( ap = (va_list)0 )
#define <span style="color:#ff6666;">va_start</span> _crt_va_start#define <span style="color:#ff6666;">va_arg</span> _crt_va_arg#define <span style="color:#ff6666;">va_end</span> _crt_va_end

然后下面上例子代码(代码来源于http://en.cppreference.com/w/cpp/utility/variadic)

#include <iostream>#include <cstdarg> void simple_printf(const char* fmt...){    va_list args;    va_start(args, fmt);     while (*fmt != '\0') {        if (*fmt == 'd') {            int i = va_arg(args, int);            std::cout << i << '\n';        } else if (*fmt == 'c') {            // note automatic conversion to integral type            int c = va_arg(args, int);            std::cout << static_cast<char>(c) << '\n';        } else if (*fmt == 'f') {            double d = va_arg(args, double);            std::cout << d << '\n';        }        ++fmt;    }     va_end(args);} int main(){    simple_printf("dcff", 3, 'a', 1.999, 42.5); }


这里解释一下,va_start就是使args得到变参的首地址。va_arg是获取下一个参数,移动sizeof(第二个参数)大小,返回下一个参数的值,最后要记得va_end。

  然后说说我在cocos2dx中看到的应用,例如在cocos2dx中Sequence创建多个动作的时候最后一个参数要填nullptr,原因就是可变参数作为临时变量压入栈中是从左往右的,然后最后一个空指针(nullptr)就在栈底,关于Sequence具体的实现我没有看,大家有兴趣自己看一下,去参数的时候可以直接用va_arg取出函数的指针,通过判断是否为nullptr来判断时候结束调用,在cocos2dx的api中我们可以看到

static Sequence * create (M m1, std::nullptr_t listEnd) static Sequence * create (M m1, M m2, std::nullptr_t listEnd) static Sequence * create (M m1, M m2, M m3, std::nullptr_t listEnd) static Sequence * create (M m1, M m2, M m3, M m4, std::nullptr_t listEnd) static Sequence * create (M m1, M m2, M m3, M m4, M m5, std::nullptr_t listEnd)
  然后关于可变参数就说到这里了


0 0