其实c语言中有很多可变函数,令人头痛,例如printf()函数,其函数原型为:
int printf( const char* format, …);
它除了有一个参数format固定以外,后面跟的参数的个数和类型是可变的(用三个点“…”做参数占位符),实际调用时可以有以下的形式:
printf("%d",i);
printf("%s",s);
printf(“the number is %d ,string is:%s”, i, s);
究竟如何写可变参数的C函数以及这些可变参数的函数编译器是如何实现,这个问题有点难度。这里我们就这个问题进行一些探讨。
看一波程序。该函数至少有一个整数参数,其后是占位符…,表示后面参数的个数不定. 在这个程序里,所有的输入参数必须都是整数,函数的功能是打印所有参数的值。
#include <<stdio.h>>
#include <<stdarg.h>>
void fun(int start, …)
{
va_list arg_ptr;
int nA =start;
int nB=0; //可变参数的数目
va_start(arg_ptr,start); //以固定参数的地址为起点确定变参的内存起始地址。
do
{
++nB;
printf(“the %d th arg: %d/n”,nB,nA); //输出各参数的值
nA = va_arg(arg_ptr,int); //得到下一个可变参数的值
} while(nA != -1);
va_end(arg_ptr);
return;
}
int main()
{
fun(100,-1);
fun(100,200,-1);
return 0;
}
可以看出这个程序用到的宏:
void va_start( va_list arg_ptr, prev_param );
type va_arg( va_list arg_ptr, type );
void va_end( va_list arg_ptr );
va在这里是variable-argument(可变参数)的意思。
这些宏定义在stdarg.h中,所以用到可变参数的程序应该包含这个头文件。 函数里首先定义一个va_list型的变量,这里是arg_ptr,这个变量是存储参数地址的指针.因为得到参数的地址之后,再结合参数的类型,才能得到参数的值。然后用va_start宏初始化⑵中定义的变量arg_ptr,这个宏的第二个参数是可变参数列表的前一个参数,即最后一个固定参数。然后依次用va_arg宏使arg_ptr返回可变参数的地址,得到这个地址之后,结合参数的类型,就可以得到参数的值。