使用可变参数,实现函数,求函数参数的平均值
可变参数要实现必须得保证参数列表中至少有一个参数,否则无法使用va_list
可变参数的实现过程: 形参实例化时给形参变量分配的地址是连续的,而且实例化时是从最右边的变量开始分配地址的,由于形参变量是分配在栈区,故第一个参数的地址是最小的,而由于给定了第一个参数,因此可以通过地址访问其他的形参变量。
通过源码可以发现这些是通过宏替换完成的,这些宏无法判断可变参数列表中的参数的数量及类型。
下面是部分源码及代码
typedef char * va_list;
#elif defined(_M_IX86)
#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
#define _crt_va_start(ap,v) ( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) )
#define _crt_va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
#define _crt_va_end(ap) ( ap = (va_list)0 )
#include <stdio.h>
#include <stdarg.h>
int average(int n, ...)
{
va_list arg;//声明va_list类型的变量,用来访问参数列表的不确定部分
int i = 0;
int sum = 0;
va_start(arg, n);//初始化变量arg,初始化后arg指向可变参数列表部分的第一个参数
for (i=0; i<n; i++)
{
sum += va_arg(arg, int);//va_arg获得这个参数的值,参数中的int指明获得参数的类型,同时arg指向下一个参数
}
return sum/n;
va_end(arg);//释放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);
return 0;
}