C語言中可變參數(shù)的使用方法示例
前言
在C語言程序編寫中我們使用最多的函數(shù)一定包括printf以及很多類似的變形體。這個函數(shù)包含在C庫函數(shù)中,定義為 int printf( const char* format, ...);
除了一個格式化字符串之外還可以輸入多個可變參量,如:
printf("%d",i); printf("%s",s); printf("the number is %d ,string is:%s", i, s);
格式化字符串的判斷本章暫且不論,下面分析一下可變參數(shù)的實現(xiàn)細節(jié)。
一,簡單實例
int simple(int num,...) { int i, result=0; va_list vl; //va_list指針,用于va_start取可變參數(shù),為char* va_start(vl,num); //取得可變參數(shù)列表中的第一個值 printf("num:%d, vl:%d\n",num,*vl); for (i = 0; i < (num-1); i++)//這里num表示可變參數(shù)列表中有多少個參數(shù) { result = va_arg(vl, int);//這里把vl往后跳過4個字節(jié)(sizeof(int)大?。┲赶蛳乱粋€參數(shù),返回的是當前參數(shù)(而非下 一個參數(shù)) printf("in for result:%d, *vl:%d\n", result, *vl);//這里打印下,可以看出,vl總是指向result后面的那個參數(shù) } 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
二.相關API介紹
可變參數(shù)列表的實現(xiàn)是由幾個宏組成的,在文件include/stdarg.h中:
va_list 定義某個變量,內(nèi)核中的定義:
typedef char *va_list;//字符指針類型
va_start(ap, type) 開始獲取可變參數(shù)列表中的第一個參數(shù)(...里面的第一個),也就是跳過第一個參數(shù)(即num):
#ifndef __sparc__ #define va_start(AP, LASTARG) \ (AP = ((char *) &(LASTARG) + __va_rounded_size (LASTARG)))//ap指向下一個參數(shù),lastarg不變 #else #define va_start(AP, LASTARG) \ (__builtin_saveregs (), \ AP = ((char *) &(LASTARG) + __va_rounded_size (LASTARG))) //跳過下第一個參數(shù),指向第二個參數(shù)內(nèi)存地址 #endif //對type向上取整 取int的整 4,然后乘上int整型4的倍數(shù) #define __va_rounded_size(TYPE) \ (((sizeof (TYPE) + sizeof (int) - 1) / sizeof (int)) * sizeof (int))
va_arg(args, int) 循環(huán)獲取到可變參數(shù)列表中的參數(shù),args指向下一個參數(shù)地址,返回的則是當前參數(shù)地址
// first=va_arg(args,int) #define va_arg(AP, TYPE) \//ap指向下一個類型的參數(shù) (AP += __va_rounded_size (TYPE), \//返回ap - sizeof(type)參數(shù),即前一個參數(shù) *((TYPE *) (AP - __va_rounded_size (TYPE)))) //對type向上取整 取int的整 4,然后乘上int整型4的倍數(shù) #define __va_rounded_size(TYPE) \ (((sizeof (TYPE) + sizeof (int) - 1) / sizeof (int)) * sizeof (int))
最后一個va_end(ap)結束標志,可能只是在程序中作為一個可變參數(shù)列表的結束標志而已(stdarg.h里面只是僅僅定義了下,沒有實現(xiàn)的代碼部分)。
三.可變參事應用是注意事項
因為va_start, va_arg, va_end等定義成宏,所以它顯得很愚蠢,可變參數(shù)的類型和個數(shù)完全在該函數(shù)中由程序代碼控制,它并不能智能地識別不同參數(shù)的個數(shù)和類型.有人會問:那么printf中不是實現(xiàn)了智能識別參數(shù)嗎?那是因為函數(shù)printf是從固定參數(shù)format字符串來分析出參數(shù)的類型,再調(diào)用va_arg的來獲取可變參數(shù)的.也就是說,你想實現(xiàn)智能識別可變參數(shù)的話是要通過在自己的程序里作判斷來實現(xiàn)的.另外有一個問題,因為編譯器對可變參數(shù)的函數(shù)的原型檢查不夠嚴格,對編程查錯不利.
如將simple可變參數(shù)該成char型指針,若存在空指針在會產(chǎn)生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; }
可變參數(shù)為char*型,當我們忘記用兩個參數(shù)來調(diào)用該函數(shù)時,就會出現(xiàn)core dump(Unix) 或者頁面非法的錯誤(window平臺).但也有可能不出錯,但錯誤卻是難以發(fā)現(xiàn),不利于我們寫出高質量的程序.
總結
到此這篇關于C語言中可變參數(shù)的使用方法示例的文章就介紹到這了,更多相關C語言可變參數(shù)用法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
C++?this原理與可變參數(shù)及友元函數(shù)友元類分步詳解用法
可變參數(shù)模板(variadic?templates)是C++11新增的強大的特性之一,它對模板參數(shù)進行了高度泛化,能表示0到任意個數(shù)、任意類型的參數(shù),這篇文章主要介紹了C++?this原理與可變參數(shù)及友元函數(shù)友元類2022-11-11