(一)源码剖析
1.va_list
typedef char * va_list
定义了一个char * 类型的va_list变量
2.va_start
#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
将n的长度变为int的整数倍
#define va_start(ap,v) (ap = (va_list)&v + _INTSIZEOF(v) )
将v取地址,然后将其类型强转成va_list类型,即转换成一个char * 类型的指针,然后加上移动_INTSIZEOF(v)个字节的地址赋给ap
3.va_arg
#define va_arg(ap,t) ( *(t *)( (ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
将指针ap指向当前位置移动_INTSIZEOF(t)个字节后的位置,然后取现在的ap所在位置的前 _INTSIZEOF(t)个字节的位置(ap移动前的原位置),并将其强制转换成t类型,然后对其解引用,取出这个地址的内容。
4.va_end
#define va_end(ap) ( ap = (va_list)0 )
将0强制转换成va_list即char *类型,赋给ap,即将ap置空
(二)实例解析
#include <stdio.h>
#include <stdarg.h>
int average(int n, ...) //n代表的是参数的个数
{
va_list arg; //char * arg
int i = 0;
int sum = 0;
va_start(arg,n);
for(i=0;i<n;i++)
{
sum += va_arg(arg,int);//sum += *(int *)((arg += _INTSIZEOF(int)) - _INTSIZEOF(int));
}
return sum/n;
va_end(arg); //arg = (char*)0
}
int main()
{
int a = 4;
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;
}