c语言中的可变参数是一个比较有意思的实现,通过将函数实现为可变参数的形式,可以使得函数可以接受1个以上的任意多个参数(不固定)。
先看一个例子:
//2.使用可变参数,实现函数,求函数参数的平均值。
#include <stdarg.h> //可变参数列表头文件
int average(int n, ...)
{
va_list arg;//vs源码中的typedef char * va_list解释,相当于定义了一个指针char *arg;
int i = 0;
int sum = 0;
va_start(arg, n);//arg=(char *)(&n+4)初始化arg为可变参数列表中第一个参数的地址
for (i = 0; i < n; i++)
{
sum += va_arg(arg, int);//sum+=(*(int *)(arg+=4)-4)
}
return sum / n;
va_end(arg);//arg=(char *)0
}
int main()
{
int a = 1;
int b = 2;
int c = 3;
int avg1 = average(2, a, c);
int avg2 = average(3, a, b, c);
printf("avg1=%d\n", avg1);
printf("avg2=%d\n", avg2);
system("pause");
return 0;
}
- 声明一个va_list类型的变量arg,它用于访问参数列表的未确定部分。
- 这个变量是调用va_start来初始化的。它的第一个参数是va_list的变量名,第二个参数是省略号前最后一个有名字的参数。初始化过程把arg变量设置为指向可变参数部分的第一个参数。
- 为了访问参数,需要使用va_arg,这个宏接受两个参数:va_list变量和参数列表中下一个参数的类型。va_arg返回这个参数的值,并使用va_arg指向下一个可变参数。
- 最后,当访问最后一个可变参数之后,我们需要调用va_end。
在vs的源码中我们可以看到:可变参数的实现过程是使用宏的分装
可变参数的限制
注意:
- 可变参数必须从头到尾逐个访问。可以在访问了几个可变参数之后半途终止,但是不可以一开始就访问参数列表中间的参数。
- 参数列表中至少有一个命名参数。如果连一个命名参数都没有,就无法使用va_start。
- 这些宏是无法直接判断每个参数的类型。
- 如果在va_arg中指定了错误的类型,其后果不可预测。