(一)源码剖析
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; }