在C语言编程中,有时候会遇到一些参数个数可变的函数,例如printf()函数,其函数原型为:
int printf(const char* format,...);
它除了有一个参数format固定以外,后面的参数的个数和类型都是可变的(用三个点“…”做参数占位符),实际调用时可以有以下的形式:
printf("%d",i);
printf("%s",s);
printf("the number is %d,string is:%s",i,s);
下面通过一个简单的函数来分析一下函数的可变参数列表。
#include<stdio.h>
#include<stdarg.h>
int max(int num,...)
{
va_list arg;
int max = 0;
int i = 0;
va_start(arg,num);
max = va_arg(arg,int);
for(i=0;i<num;i++)
{
int temp = va_arg(arg,int);
if(temp > max)
{
max = temp;
}
}
va_end(arg);
return max;
}
int main()
{
int n = max(3,4,5,6);
printf("%d\n",n);
return 0;
}
1.
在程序中将用到以下这些宏:
void va_start(va_list arg_ptr,prev_param);
type va_arg(va_list arg_ptr,type);
void va_end(va_lst arg_ptr);
va 在这里的意思是variable-argument(可变参数);
arg_ptr 是指向可变参数列表的指针;
prev_param 则指可变参数表的前一个固定参数;
type 为可变参数的类型。
va_list 也是一个宏,其定义为typedef char *va_list,实质上是一个char型指针。char型指针的特点是++、–操作的结果是增1或者减1,因为sizeof(char)=1,与之不同的是int等其他类型指针的++、–操作的结果是增sizeof(type),而且sizeof(type)大于1.
这些宏定义在stdarg.h中,所以用到可变参数的程序应该包含这个头文件。
2.
函数里首先定义一个va_list型的变量arg,它用于访问参数列表的未确定部分,注意arg是一个指针,它指向n的后面。因为va _ list是系统自定义的,所以我们一般不要直接取值,而是调用va _ arg(arg,int),它的返回值是arg指针指向的内容,并且将arg下移一位:
va_list arg;
在进行参数调用时,会从右往左把参数压入栈中,对于max(3,4,5,6)的调用,其栈内容如图所示:
3.
然后用va_start初始化(2)中的arg,把arg设置为指向可变参数部分的第一个参数n:
va_start(arg,n);
4.
设定结束条件,这里的结束条件就是判断i是否小于num。注意被调的函数在调用时是不知道可变参数的正确数目的,程序员必须自己在代码中指明结束条件。
va_end(arg);
需要注意的是,在退出main函数之前,必须调用va_end,否则可能会导致不可预料的结果。