我最近對可變參數列表十分感興趣,因爲它可以接受不同數目的參數。讓我們先看一下它的聲明:
void printf (const char *format,...)
我們通過一個具體的例子來理解它:
#include <stdio.h> #include <stdarg.h> int average (int val, ...) { int i = 0; int sum = 0; va_list arg;//定義了一個char 類型的指針 va_start(arg,val);//讀取所有的參數,從頭開始,後讀取你所給的參數個數,並且把參數放到指針中 for (i = 0; i < val; i++) { sum += va_arg(arg,int); } va_end(arg);//結束讀取 return sum / val; } int main() { int ret = 0; ret = average(5,1,2,3,4,5); printf("%d\n",ret); return 0; }
可變參數列表是通過宏-stdarg來實現的,這個頭文件中聲明瞭一個類型va_list和三個宏va_start,va_arg,va_end,通過右擊並轉到定義處,我們可以發現va_list其實就是一個char *的重命名,函數中是這樣寫的,typedef char * va_list,這就非常好理解了;對於va_start,va_arg和va_end三個宏我們分別轉到定義處可以發現va_start其實是這樣的:,而_crt_va_start依然是由define定義的標識符,其實應該是這樣( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) );而va_arg和va_start類似也是由define定義的標識符,它的原型應該是這樣的:#define _crt_va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) ),ap就是指向我的可變參數列表中的第一個參數,通過它可以找到之後的參數;對於va_end它的原型應該是這樣的#define _crt_va_end(ap) ( ap = (va_list)0 ),用於結束va_list說指向的參數列表,當訪問完最後一個可變參數之後,我們就需要調用va_end來結束指向的參數列表