最近在看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)
然后关于可变参数就说到这里了