您好,登錄后才能下訂單哦!
這期內容當中小編將會給大家帶來有關如何在C語言中使用可變參數,文章內容豐富且以專業的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。
int simple(int num,...) { int i, result=0; va_list vl; //va_list指針,用于va_start取可變參數,為char* va_start(vl,num); //取得可變參數列表中的第一個值 printf("num:%d, vl:%d\n",num,*vl); for (i = 0; i < (num-1); i++)//這里num表示可變參數列表中有多少個參數 { result = va_arg(vl, int);//這里把vl往后跳過4個字節(sizeof(int)大小)指向下一個參數,返回的是當前參數(而非下 一個參數) printf("in for result:%d, *vl:%d\n", result, *vl);//這里打印下,可以看出,vl總是指向result后面的那個參數 } va_end(vl);//結束標志 return result; } int main(int argc, char **argv) { int num = argc; int i = 0; simple(5,1,2,3,4,5); return 1; }
運行結果如下:
book@book-desktop:~/own$ ./varlist
num:5, vl:1
in for result:1, *vl:2
in for result:2, *vl:3
in for result:3, *vl:4
in for result:4, *vl:5
可變參數列表的實現是由幾個宏組成的,在文件include/stdarg.h中:
va_list 定義某個變量,內核中的定義:
typedef char *va_list;//字符指針類型
va_start(ap, type) 開始獲取可變參數列表中的第一個參數(...里面的第一個),也就是跳過第一個參數(即num):
#ifndef __sparc__ #define va_start(AP, LASTARG) \ (AP = ((char *) &(LASTARG) + __va_rounded_size (LASTARG)))//ap指向下一個參數,lastarg不變 #else #define va_start(AP, LASTARG) \ (__builtin_saveregs (), \ AP = ((char *) &(LASTARG) + __va_rounded_size (LASTARG))) //跳過下第一個參數,指向第二個參數內存地址 #endif //對type向上取整 取int的整 4,然后乘上int整型4的倍數 #define __va_rounded_size(TYPE) \ (((sizeof (TYPE) + sizeof (int) - 1) / sizeof (int)) * sizeof (int))
va_arg(args, int) 循環獲取到可變參數列表中的參數,args指向下一個參數地址,返回的則是當前參數地址
// first=va_arg(args,int) #define va_arg(AP, TYPE) \//ap指向下一個類型的參數 (AP += __va_rounded_size (TYPE), \//返回ap - sizeof(type)參數,即前一個參數 *((TYPE *) (AP - __va_rounded_size (TYPE)))) //對type向上取整 取int的整 4,然后乘上int整型4的倍數 #define __va_rounded_size(TYPE) \ (((sizeof (TYPE) + sizeof (int) - 1) / sizeof (int)) * sizeof (int))
最后一個va_end(ap)結束標志,可能只是在程序中作為一個可變參數列表的結束標志而已(stdarg.h里面只是僅僅定義了下,沒有實現的代碼部分)。
因為va_start, va_arg, va_end等定義成宏,所以它顯得很愚蠢,可變參數的類型和個數完全在該函數中由程序代碼控制,它并不能智能地識別不同參數的個數和類型.有人會問:那么printf中不是實現了智能識別參數嗎?那是因為函數printf是從固定參數format字符串來分析出參數的類型,再調用va_arg的來獲取可變參數的.也就是說,你想實現智能識別可變參數的話是要通過在自己的程序里作判斷來實現的.另外有一個問題,因為編譯器對可變參數的函數的原型檢查不夠嚴格,對編程查錯不利.
如將simple可變參數該成char型指針,若存在空指針在會產生coredump
void simple(int i, ...) { va_list arg_ptr; char *s=NULL; va_start(arg_ptr, i); s=va_arg(arg_ptr, char*); va_end(arg_ptr); printf("%d %s\n", i, s); return; }
可變參數為char*型,當我們忘記用兩個參數來調用該函數時,就會出現core dump(Unix) 或者頁面非法的錯誤(window平臺).但也有可能不出錯,但錯誤卻是難以發現,不利于我們寫出高質量的程序.
上述就是小編為大家分享的如何在C語言中使用可變參數了,如果剛好有類似的疑惑,不妨參照上述分析進行理解。如果想知道更多相關知識,歡迎關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。