關於c++可變參數的理解

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


發佈了23 篇原創文章 · 獲贊 9 · 訪問量 2萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章