最近在看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)
然後關於可變參數就說到這裏了