C语言编程中有时会遇到一些参数个数可变的函数,例如printf函数的原型为
int printf (const char*format,...)
它除了有一个参数format固定以外,后面的参数个数和类型都是可变的(用"..."作为参数占位符),在实际调用时有以下的形式:
printf("%d\n",i);
print("%s",s);
printf("the number is %d.string is %s",i,s);
以一个简单的可变参数的C函数为例,该函数至少有一个整数参数,其后是占位符"...",表示后面参数的个数不定。在这个函数里,所有的输入参数必须都是整数,函数的功能只是打印所有参数的值。函数代码如下
#include<stdio.h>
#include<stdarg.h>
void simple_va_fun(int start,...)
{
va_list arg_ptr;
int nArgValue = start;
int nArgCount = "0"; //可变参数的数目
va_start(arg_ptr,start); //以固定参数的地址以起点确定变参的内存起始地址
do
{
++nArgCount;
printf("the %d th arg:%d",nArgCount,nArgValue );
nArgValue = va_arg(arg_ptr,int);
}while(nArgValue != -1);
return;
}
int main()
{
simple_va_fun(100,-1);
simple_va_fun(100,200,-1);
return 0;
}
从这个函数的实现可以看到,我们使用可变参数应该包括以下步骤。
(1)在程序中将用到以下这些宏
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"是可变参数的意思。这些宏定义在stdarg.h中,所以用到可变参数的程序应该包含这个头文件
(2)在函数中定义一个va_list型的变量,此处为arg_ptr。这个变量是存储参数地址的指针,因为在得到参数的地址之后,再结合参数的类型,才能得到参数的值。
(3)在va_start()宏初始化(2)中定义的变量arg_ptr,这个宏的第二个参数是可变参数列表的前一个参数,即最后一个固定参数
(4)依次用va_arg宏使arg_ptr返回可变参数的地址,在得到这个地址后,结合参数的类型,就可以得到参数的值
(5)设定结束条件,本例的条件就是判断参数值是否为-1,注意,函数在被调用时是不知道可变参数的正确数目的,程序员必须自己在代码中指明结束条件。至于为什么被调函数不知道参数的数目,在看完这几个宏的内部实现机制后,自然就会明白