在定義可變參數的函數之前,先來理解一下函數參數的傳遞原理:
1、函數參數是以棧這種數據結構來存取的,在函數參數列表中,從右至左依次入棧。
2、參數的內存存放格式:參數的內存地址存放在內存的堆棧段中,在執行函數的時候,從最後一個(最右邊)參數開始入棧。因此棧底高地址,棧頂低地址,舉個例子說明一下:
void test(int a, float b, char c);
那麼,在調用test函數的時候,實參char c先進棧,然後是float b,最後才是int a,因此在內存中變量的存放次序是c->b->a,因為從理念上來說,我們只要探測到任意一個變量的地址,並且知道其它變量的類型,通過指針移位運算,就可以順籐摸瓜找到其它的輸入變量。
實現一個可變參數的函數,需要用到以下幾個宏:
typedef char* va_list; // 用於聲明一個指向參數列表的字符型指針變量 void va_start(va_list ap, prev_param); // 第一個參數為指向可變參數字符指針變量,第二個參數是可變參數的第一個參數,通常用於指定可變參數列表中參數的個數 void va_arg(va_list ap, type); // 第一個參數為指向可變參數字符指針變量,第二個參數是可變參數的數據類型 void va_end(va_list ap);// 將存放可變參數字符串的變量清空(賦值為NULL)3、示例:求N個數的和
int sum(int count, ...) { int sum = 0; int i; va_list ap; va_start(ap, count); for (i = 0; i < count; ++i) { sum += va_arg(ap, int); } va_end(ap); return sum; }
int main(int argc, const char * argv[]) { int ret = sum(5, 1, 2, 3, 4, 5); printf("sum: %d\n",ret); }4、示例:標准庫可變參數的相關函數使用
void test(int count,...) { va_list ap; va_start(ap, count); vprintf("%d,%d,%d\n", ap); // 格式化輸出可變參數的值 char buff[1024]; vsprintf(buff, "a=%d,b=%d,c=%d\n", ap); // 將可變參數列表的值格式化輸出到緩沖區中 printf("%s\n",buff); vfprintf(stdout, "a=%d,b=%d,c=%d\n", ap); // 將可變參數列表的值打印到標准輸出中 // 可變參數傳入的是int類型的地址 vsscanf("10,30,40", "%d,%d,%d", ap); // 依次將格式化字符串的值,輸入到可變參數變量中 vsnprintf(buff, 30, "a=%d,b=%d,c=%d", ap); // 將可變參數列表的值,格式化輸出指定長度(30個字符)的字符串到緩沖區中 printf("vsnprintf=%s\n",buff); va_end(ap); }