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ù)的實(shí)現(xiàn)細(xì)節(jié)。
一,簡單實(shí)例
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ù),返回的是當(dāng)前參數(shù)(而非下 一個參數(shù)) printf("in for result:%d, *vl:%d\n", result, *vl);//這里打印下,可以看出,vl總是指向result后面的那個參數(shù) } va_end(vl);//結(jié)束標(biāo)志 return result; } int main(int argc, char **argv) { int num = argc; int i = 0; simple(5,1,2,3,4,5); return 1; }
運(yùn)行結(jié)果如下:
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
二.相關(guān)API介紹
可變參數(shù)列表的實(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ù)地址,返回的則是當(dāng)前參數(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)結(jié)束標(biāo)志,可能只是在程序中作為一個可變參數(shù)列表的結(jié)束標(biāo)志而已(stdarg.h里面只是僅僅定義了下,沒有實(shí)現(xiàn)的代碼部分)。
三.可變參事應(yīng)用是注意事項(xiàng)
因?yàn)関a_start, va_arg, va_end等定義成宏,所以它顯得很愚蠢,可變參數(shù)的類型和個數(shù)完全在該函數(shù)中由程序代碼控制,它并不能智能地識別不同參數(shù)的個數(shù)和類型.有人會問:那么printf中不是實(shí)現(xiàn)了智能識別參數(shù)嗎?那是因?yàn)楹瘮?shù)printf是從固定參數(shù)format字符串來分析出參數(shù)的類型,再調(diào)用va_arg的來獲取可變參數(shù)的.也就是說,你想實(shí)現(xiàn)智能識別可變參數(shù)的話是要通過在自己的程序里作判斷來實(shí)現(xiàn)的.另外有一個問題,因?yàn)榫幾g器對可變參數(shù)的函數(shù)的原型檢查不夠嚴(yán)格,對編程查錯不利.
如將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*型,當(dāng)我們忘記用兩個參數(shù)來調(diào)用該函數(shù)時,就會出現(xiàn)core dump(Unix) 或者頁面非法的錯誤(window平臺).但也有可能不出錯,但錯誤卻是難以發(fā)現(xiàn),不利于我們寫出高質(zhì)量的程序.
總結(jié)
到此這篇關(guān)于C語言中可變參數(shù)的使用方法示例的文章就介紹到這了,更多相關(guān)C語言可變參數(shù)用法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C語言算法練習(xí)之?dāng)?shù)組求素?cái)?shù)
這篇文章主要為大家介紹了C語言算法練習(xí)中數(shù)組求素?cái)?shù)的實(shí)現(xiàn)方法,文中的示例代碼講解詳細(xì),對我們學(xué)習(xí)C語言有一定幫助,需要的可以參考一下2022-09-09C++?this原理與可變參數(shù)及友元函數(shù)友元類分步詳解用法
可變參數(shù)模板(variadic?templates)是C++11新增的強(qiáng)大的特性之一,它對模板參數(shù)進(jìn)行了高度泛化,能表示0到任意個數(shù)、任意類型的參數(shù),這篇文章主要介紹了C++?this原理與可變參數(shù)及友元函數(shù)友元類2022-11-11詳解桶排序算法的思路及C++編程中的代碼實(shí)現(xiàn)
桶排序即是先把每個桶中的元素進(jìn)行排序然后遍歷桶依次列出元素的算法,桶排序在元素較少的情況下很高效,以下我們就來詳解桶排序算法的思路及C++編程中的代碼實(shí)現(xiàn):2016-07-07vc6.0中c語言控制臺程序中的定時技術(shù)(定時器)
這篇文章主要介紹了vc6.0中c語言控制臺程序中的定時技術(shù)(定時器),需要的朋友可以參考下2014-04-04