可变參数列表
函数原型中,列出了函数期望接受的參数,但函数仅仅能显示固定数目的參数。让一个函数在不同的时候接受不同的数目的參数是不是能够呢?答案是肯定的(printf就是一个可变參函数),但存在一些限制。
int avarage(int val, int v1, int v2, int v3,int v4, int v5) { floatsum = v1; if(val>= 2) sum+= v2; if(val>= 3) sum+= v3; if(val>= 4) sum+= v4; if(val>= 5) sum+= v5; returnsum / val; }
显然这是一个差劲的样例,这个函数存在几个问题:首先,它不正确參数的数量进行測试,无法检測參数过多的这样的情况。只是这个问题easy解决,简单加上測试就是了。其次,函数无法处理超过5个的值。要解决问题,你仅仅有在已经非常臃肿的代码中再添加一些类似的代码。另外还存在一个更为严重的问题:就是实參数与形參的相应不一定准确。
调用的时候avarage(3,1,2,3);
可是这样会有一个错误。
error C2660: 'avarage' : function doesnot take 3 parameters
为了解决上述的问题:
C语言中引入了可变參数列表的概念。
stdarg宏
可变參数列表是通过宏来实现的,这些宏定义于stdarg.h头文件里,它是标准库的一部分。
例如以下程序:
//指定数量的值的平均值 2 #include<stdarg.h> 3 float(int values,...) 4 { 5 va_list var_arg; 6 int count; 7 float sum=0; 8 9 var_start(var_arg,n_values);//准备訪问可变參数 10 for(count=0;count<n_values;count+=1) //加入取自可变參数表的值 11 { 12 sum+=var_varg(var_arg,int); 13 } 14 var_end(var_arg); //完毕处理可变參数 15 return sum/n_values; 16 }
当中有几个变量须要说明一下。va_list、va_start()、va_end和va_arg。
Va_list:该类型变量用来訪问可变參数,实际上就是指针。
Va_start():是一个宏,用来获取參数列表中的參数,使vl指向第一个可变參数,使用完成后调用va_end()结束。
va_end:也是一个宏,用来结束va_start()的调用。
va_arg:宏,用来获參数列表中的取下一个值。
声明一个va_list类型的变量arg,它用于訪问參数列表的未确定部分。这个变量是调用va_start来初始化的。它的第一个參数是va_list的变量名,第2个參数是省略号前最后一个有名字的參数。初始化过程把var_arg变量设置为指向可变參数部分的第一个參数。
为了訪问參数,须要使用va_arg,这个宏接受两个參数:va_list变量和參数列表中下一个參数的类型。在这个样例中全部的可变參数都是整型。va_arg返回这个參数的值,并使用var_arg指向下一个可变參数。
最后,当訪问完成最后一个可变參数之后,我们须要调用va_end。
可变參数的限制
注意,可变參数必须从头到尾逐个訪问。假设你在訪问了几个可变參数之后想半途终止,这是能够的,可是,假设你想一開始就訪问參数列表中间的參数,那是不行的。
1.參数列表中至少有一个命名參数。假设连一个命名參数都没有,就无法使用va_start。
2.这些宏是无法直接推断实际存在參数的数量。
3.这些宏无法推断每一个參数的是类型。
4.假设在va_arg中指定了错误的类型,那么其后果是不可预測的。