VA_LIST 是在C語言中解決變參問題的一組宏,定義在<stdarg.h>標頭檔下。
VA_LIST的用法:
(1)首先在函數裡定義一具VA_LIST型的變數,這個變數是指向參數的指標
(2)然後用VA_START宏初始設定變數剛定義的VA_LIST變數,這個宏的第二個參數是第一個可變參數的前一個參數,是一個固定的參數。
(3)然後用VA_ARG返回可變的參數,VA_ARG的第二個參數是你要返回的參數的類型。
(4)最後用VA_END宏結束可變參數的擷取。然後你就可以在函數裡使用第二個參數了。如果函數有多個可變參數的,依次調用VA_ARG擷取各個參數。
#include<stdio.h>#include <stdarg.h>void arg_cnt(int cnt, ...){ int value=0; int i=0; int arg_cnt=cnt; va_list arg_ptr; va_start(arg_ptr, cnt); for(i = 0; i < cnt; i++) { value = va_arg(arg_ptr,int); printf("value%d=%d\n", i+1, value); } va_end(arg_ptr);}int main(void){ arg_cnt(5,1,2,3,4,5); return 0;}
運行上面的程式:
value1=1
value2=2
value3=3
value4=4
value5=5
你發現這個程式有什麼不方便的地方了嗎?
對!就是要在第一個參數指定可變參數的數量,就是arg_cnt函數中的cnt變數,
如果我在main中調用
arg_cnt(6,1,2,3,4,5);
那麼結果就是沒法預測了。
下面再介紹一個vsnprint函數,該函數支援可變參數,說明如下:
標頭檔:#include <stdarg.h>函式宣告:int vsnprintf(char *buffer, size_t max_count, const char *format, va_list vArgList);參數說明:char *buffer [out],把產生的格式化的字串存放在這裡.size_t max_count [in], buffer可接受的最大位元組數,防止產生數組越界.const char *format [in], 格式化字串va_list vArgList [in], va_list變數. va:variable-argument:可變參數用法類似於vsprintf,只不過加了max_count的限制.
傳回值說明:如果成功調用此函數,返回寫到buffer中的字元的個數(不包括結尾的'\0')。snprintf和vsnprintf函數不能夠寫多於size大小(包括結尾的'0')的位元組數。如果輸出因為以上原因被截斷,返回成功寫入buffer的字元數(不包括結尾的'\0'),如果有足夠的記憶體空間的話。所以,如果傳回值等於size或者大於size,表示輸出到buffer的字元被截斷,如果輸出過程中遇到錯誤,則返回一個負數。
#include<stdio.h>#include <stdarg.h>#define bufsize 80char buffer[bufsize];int vspf(char *fmt, ...){ va_list argptr; int cnt; va_start(argptr, fmt); cnt = vsnprintf(buffer,bufsize ,fmt, argptr); va_end(argptr); return(cnt);}int main(void){ int inumber = 30; float fnumber = 90.0; char string[4] = "abc"; int cnt = 0; cnt = vspf("%d %f %s", inumber, fnumber, string); printf("%s\n", buffer); printf("cnt=%d\n", cnt); return 0;}
執行結果如下:30 90.000000 abccnt=16