C语言中可变参数是一个比较有意思的实现,通过将函数实现为可变参数的形式,可以使得函数可以接受一个以上任意多个参数(不固定),例如printf()函数等
函数参数进栈顺序从右往左,如下图从上到下;
在上图中我们要读取可变部分,怎么读取呢?
首先指针char *指向最下面,对num取地址,并且强转为char *,并且加上num类型的大小,这时指针指向a,然后对指针强转为int*解引用,就可以得到a的内容,以此类推。
所以要使用可变参数列表需满足以下条件:
(1)必须知道每个参数的类型; 由输入%d,%c等可知
(2)必须要有一个具体的明确的参数;
(3)明确知道多少个参数;
实例1:
实现一个函数可以求任意个参数的平均值; |
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include<stdarg.h>
int my_avg(int n, ...) //n表示几个数
{
va_list arg; //char *arg
va_start(arg, n); //让char *指向可变部分
//相当于arg = (char*)n;arg+=sizeof(num);
int sum = 0;
int count = n;
//接下来循环表示提取每个可变部分的内容
while (count){
sum += va_arg(arg, int);
//相当于 *((int *)arg);arg+=sizeof(int)
count--;
}
va_end(arg); //arg = NULL 用于结束
return sum / n;
}
int main()
{
int a = 10;
int b = 20;
int c = 30;
int avg1 = my_avg(2, a, b);
int avg2 = my_avg(3, a, b, c);
printf("avg1:%d\navg2:%d\n", avg1, avg2);
system("pause");
return 0;
}
- 声明一个va_list(相当于char*)类型的变量arg,它用于访问参数列表中未确定部分(可变部分)。
- arg变量是调用va_start来初始化的。他的第一个参数是va_list的变量名,第二个参数是省略号前最后一个有名字的参数。初始化过程把arg变量设置为指向可变参数部分的第一个参数。红色部分why?为了方便直接定位可变部分。
- 为了访问参数,需要使用va_arg,这个宏接受两个参数:va_list变量和参数列表中下一个参数的类型。在这个例子中所有的可变参数都为整形。va_arg返回这个参数的值,并使用va_arg指向下一个可变参数。
- 最后,当访问完最后一个可变参数之后,我们需要调用va_end。
【可变参数的限制】
注意:
- 可变参数必须从头到尾逐个访问。如果你在访问了几个可变参数之后想半途终止,这个是可以的。但是,如果你想一开始就访问可变现参数列表中间的某个,这是不可以的。
- 参数列表中至少有一个命名参数。如果连一个命名参数都没有,就无法使用va_start。
- 这些宏无法直接判断实际从在参数的数量。
- 这些宏无法判断每个参数的类型。
- 如果在va_arg中指定了错误的类型,后果不可预测。
实例2:
利用可变参数列表求几个数的最大值
int int_max(int n, ...)
{
va_list arg;
va_start(arg, n);
int max = va_arg(arg, int);
int i = 2;
int mid = 0;
while (i <= n)
{
mid = va_arg(arg, int);
if (max < mid)
{
max = mid;
}
i++;
}
return max;
va_end(arg);
}
int main()
{
int a = 1, b = 2, c = 6, d = 72, e = 7;
printf("max = %d\n", int_max(5, a, b, c, d, e));
system("pause");
return 0;
}