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 simple_va_fun(int start, ...){
va_list arg_ptr;
int nArgValue =start;
int nArgCout="0"; //可变参数的数目
va_start(arg_ptr,start); //以固定参数的地址为起点确定变参的内存起始地址do
{++nArgCout;
printf("the %d th arg: %d",nArgCout,nArgValue); //输出各参数的值
nArgValue = va_arg(arg_ptr,int); //得到下一个可变参数的值
} while(nArgValue != -1);
return;}
int main(int argc, char* argv[]){
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在这里是variable-argument(可变参数)的意思,这些宏定义在stdarg.h中,所以用到可变参数的程序应该包含这
个头文件。
(2)函数里首先定义一个va_list型的变量,这里是arg_ptr,这个变量是存储参数地址的指针,因为得到参数的地址
之后,再结合参数的类型,才能得到参数的值。
(3)然后用va_start宏初始化(2)中定义的变量arg_ptr,这个宏的第二个参数是可变参数列表的前一个参数,即最
后一个固定参数。
(4)然后依次用va_arg宏使arg_ptr返回可变参数的地址,得到这个地址之后,结合参数的类型,就可以得到参数的值。
(5)设定结束条件,这里的条件就是判断参数值是否为-1。注意被调的函数在调用时是不知道可变参数的正确数目
的,程序员必须自己在代码中指明结束条件。至于为什么它
不会知道参数的数目,在看完这几个宏的内部实现机制后,自然就会明白。
对可变参函数的一些总结:
1.标准C库的中的三个宏的作用只是用来确定可变参数列表中每个参数的内存地址,编译器是不知道参数的实际数目的。
2.在实际应用的代码中,程序员必须自己考虑确定参数数目的办法,如
(1)在固定参数中设标志-- printf函数就是用这个办法。
(2)在预先设定一个特殊的结束标记,就是说多输入一个可变参数,调用时要将最后一个可变参数的值设置成这个
特殊的值,在函数体中根据这个值判断是否达到参数的结尾。
无论采用哪种办法,程序员都应该在文档中告诉调用者自己的约定。
3.实现可变参数的要点就是想办法取得每个参数的地址,取得地址的办法由以下几个因素决定:
(1)函数栈的生长方向
(2)参数的入栈顺序
(3)CPU的对齐方式
(4)内存地址的表达方式
结合源代码,我们可以看出va_list的实现是由④决定的,_INTSIZEOF(n)的引入则是由③决定的,他和①②又一起决
定了va_start的实现,最后va_end的存在则是良好编程风格
的体现,将不再使用的指针设为NULL,这样可以防止以后的误操作。
4.取得地址后,再结合参数的类型,程序员就可以正确的处理参数了。