我们通过下面这个简单的例子来了解一下C语言可变参数列表的使用:
#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
int get_max(int n,...)
{
va_list arg; //声明一个va_list变量arg,用于访问参数列表未确定的部分
va_start(arg, n); //第一个参数是va_list变量,第二个参数是省略号前最后一个有名字的参数
int i = 0; //初始化过程把arg变量设置为指向可变参数部分的第一个参数
int temp = 0;
int max = va_arg(arg,int); //访问可变参数,传入va_arg的第一个参数是va_list变量,第二个是参数列表中下一个参数的类型
for (i = 0; i < n-1; i++)
{
temp = va_arg(arg,int);
if (max < temp)
{
max = temp;
}
}
return max;
va_end(arg); //访问完最后一个参数后,调用va_end
}
int main()
{
int a = 1;
int b = 4;
int c = 3;
printf("%d\n",get_max(3,a,b,c));
system("pause");
return 0;
}
一些需要注意的点:
1、可变参数必须从头至尾逐个访问,不可直接访问中间的参数。
2、参数列表中至少有一个命名参数。否则无法使用va_start。
3、这些宏无法判断参数的数量和类型。
4、如果va_arg中指定了错误的类型,结果无法预测。
我们来看一下可变参数列表的源码:
#ifdef _M_ALPHA
typedef struct {
char *a0; /* pointer to first homed integer argument */
int offset; /* byte offset of next parameter */
} va_list;
#else
typedef char * va_list; //va_list其实就是char *
#endif
#define _VA_LIST_DEFINED
#endif
#ifdef _M_IX86
#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1))
#define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) )
#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
#define va_end(ap) ( ap = (va_list)0 )