关于可变参数列表源码的剖析以 求未知参数个数的平均值函数为例:
源码如下:
#include<stdio.h>
#include<stdarg.h>
int average(int n,...);
#pragma warning(disable:4996)
int main()
{
int ret=average(4, 1,2,3,4);
printf("%d\n",ret);
return 0;
}
int average(int n,...)
{
int sum=0;
int i=0;
va_list arg;//VA_LIST 是在C语言中解决变参问题的一组宏,在函数里定义一具VA_LIST型的变量,这个变量是指向参数的指针
va_start(arg,n);//VA_START宏,获取可变参数列表的第一个参数的地址(arg是类型为va_list的指针,n是可变参数最左边的参数)
for(i=0;i<n;i++)
{
sum+=va_arg(arg,int);//VA_ARG宏,获取可变参数的当前参数,返回指定类型并将指针指向下一参数(int 参数描述了当前参数的类型)
}
va_end(arg);//VA_END宏,清空va_list可变参数列表
return sum/n;
}
#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
/*sizeof(n)=4(00000100),sizeof(int)=4(00000100),(sizeof(n)+sizeof(int)-1)=(00000111)
~(sizeof(int)-1)=(11111100),(00000111&11111100)=(000000100),则INTSIZEOF(n)功能是就n的长度化为int 的长度的整数倍*/
2.va_list arg // 相当于 char *arg
2.#define va_start(arg,n) ( arg = (va_list)&n + _INTSIZEOF(n) )
/*上面的代码可以转换成arg=(char *)&n+4,意思是将&n强制类型转换成char *之后向后跳4个自己,也就是跳过n,arg指向第二个参数*/
3.#define va_arg(arg,int) ( *(int *)((arg += _INTSIZEOF(int)) - _INTSIZEOF(int)) )
/*上面代码可以转换成*(int*)((arg+=4-4))意思是arg当前指向内存空间里的值,而arg=arg+4,表示的是指向跳向一个int类型大小后的地址*/
4.#define va_end(arg) ( arg = (va_list)0 )
/*上面代码可以转换成(arg=(char *)0)/*arg是空指针,表示结束。