可变参数列表是通过函数实现为可变参数的形式,可以接受1个以上的任意多个参数,可变参数列表要求至少有1个确定的参数。
我们用的第一个可变参数列表的函数就是 printf函数
例:
int printf(const char *format[,argument]...)
int average(int n,...)
可变参数列表是通过函数实现的,故函数传参时要形成临时变量且保存在栈桢中,函数传参是从右往左依次保存在栈桢上的,我们可以找到可变参数列表中第一个确定的参数的地址,接下来我们让该地址加上该数据类型大小就可以找到下一个参数的地址,对此时的地址解引用就可以拿到第二个参数……以此类推,我们就可以找出可变参数列表的所有参数。
实现一个函数可以求任意个参数的平均值
#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
int average(int n, ...)
{
va_list arg;
int i = 0;
int sum = 0;
va_start(arg, n);
for (i = 0; i < n; i++)
{
sum += va_arg(arg, int);
}
return sum / n;
va_end(arg);
}
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理解成char *类型;
va_start(arg,n)是宏函数,它的作用是:让arg指针指向可变参数列表的不确定部分n,即让 arg = (&n+sizeof(n)),此时的arg指向的是第一个确定参数后面的不确定参数;
va_arg(arg,int)是宏函数,它的作用是:通过类型提取指定的参数,并且让指针arg + sizeof(int)指向下一个未知参数的地址;
va_end(arg)是宏函数,当访问完最后一个可变参数后,需要调用宏函数va_end()。
注意:
·可变参数必须从头到尾逐个访问。若在访问几个可变参数后想终止,这是允许的,但是,若一开始就想访问可变参数列表中间的参数,那是不行的。
·参数列表中至少要有一个命名参数,否则就无法使用va_start,即无法找到可变参数。
·宏函数不能判断参数的数量。
·宏函数不能判断参数的类型。
·若在va_arg中指定了错误的类型,后果不可预测。