一直很好奇,printf()函数怎么可以带随意数目的参数,最近逛博客终于了解了。原来C语言是支持函数参数的可变性的啊。先上代码。
#include <stdarg.h>
void f(int num, ...)
{
va_list valist;
int i;
va_start(valist, num);
for(i = 0; i < num; i++)
printf("%d/n", va_arg(valist, int));
va_end(valist);
}
int main()
{
f(5, 4, 3, 2, 1, 0);
return 0;
}
结果输出:
4
3
2
1
0
第一次看到函数形式参数是省略号的。很是新颖,省略号代表了“等等”,“更多”的意思吧。
在"stdarg.h"中,这些函数是这么定义的:
typedef char * va_list;
#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 )
定义_INTSIZEOF(n)主要是为了某些需要内存的对齐的系统.C语言的函数是从右向左压入堆栈的。
其他函数解释如下:
va_list 可以理解为参数列表,其类型是地址(char*)。
va_sart(): 得到第一个参数,即参数个数,并把指针移向后一个参数。
va_arg(valist, int): 得到下一个函数参数,其中int指定参数类型(可以是任意类型),并把指针移向后一个参数。
va_end(): 置参数列表为空。(只是指针设为空)。
考虑到调用函数时,实参都是按序压栈的,我们可以根据地址指针和实参类型(主要是获取size)获得实参的值
。
但是这种直接对堆栈数据进行访问的做法,破坏了C语言的抽象性,毕竟直接通过指针访问堆栈是危险的。但是在却当的时候使用,却显得十分的便利。我们在使用时应随机应变。
可变实参数量的原理其实显而易见。只是第一次见到这样的语法和用法,感觉新颖,故此记录并分享一下。