我最近对可变参数列表十分感兴趣,因为它可以接受不同数目的参数。让我们先看一下它的声明:

void printf (const char *format,...)

   我们通过一个具体的例子来理解它:

#include <stdio.h>
#include <stdarg.h>
int average (int val, ...)
{
	int i = 0;
	int sum = 0;
	va_list arg;//定义了一个char 类型的指针
	va_start(arg,val);//读取所有的参数,从头开始,后读取你所给的参数个数,并且把参数放到指针中
	for (i = 0; i < val; i++)
	{
		sum += va_arg(arg,int);
	}
	va_end(arg);//结束读取
	return sum / val;
}
int main()
{
	int ret = 0;
	ret = average(5,1,2,3,4,5);
	printf("%d\n",ret);
	return 0;
}

   可变参数列表是通过宏-stdarg来实现的,这个头文件中声明了一个类型va_list和三个宏va_start,va_arg,va_end,通过右击并转到定义处,我们可以发现va_list其实就是一个char *的重命名,函数中是这样写的,typedef char * va_list,这就非常好理解了;对于va_start,va_arg和va_end三个宏我们分别转到定义处可以发现va_start其实是这样的:,而_crt_va_start依然是由define定义的标识符,其实应该是这样( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) );而va_arg和va_start类似也是由define定义的标识符,它的原型应该是这样的:#define _crt_va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) ),ap就是指向我的可变参数列表中的第一个参数,通过它可以找到之后的参数;对于va_end它的原型应该是这样的#define _crt_va_end(ap) ( ap = (va_list)0 ),用于结束va_list说指向的参数列表,当访问完最后一个可变参数之后,我们就需要调用va_end来结束指向的参数列表