可变参数列表是通过宏-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来结束指向的参数列表;
下面我们来看一个由可变参数实现的求平均数的例子:
#include<stdaarg.h>
#include<stdio.h>
int average
(int val,...)
{
int i=0;
int sum = 0;
va_list arg;//
创建一个char *型变量,记录参数的起始地址
va_start(arg,val); // ,va_start()初始化变量arg;val代表可变参数的前一个变量
for(i=0;i<val;i++)
{
sum +=va_arg(arg,int);
}
va_end(arg)
; //
当访问完最后一个参数后 调用va_end()
return sum/val;
}
int main()
{
int ret = average(1, 2,3);
printf("%d\n"ret);
return 0;
}
可变参数列表虽然可以让一个函数可以在不同的情况下接受不同数目的参数,但在使用时还存在一些限制:
1、可变参数列表必须从头到尾逐个访问,如果你访问了几个参数后想中途终止,这是可以的;但是如果你想一开始就访问参数列表中间的某个参数这是不可以的。
2、参数列表中至少要有一个参数,否则无法使用va_start 。
3、va_start、 va_arg 和va_end 这三个宏是无法判断参数类型的