以下均在vs2008 中实现
通过将函数实现为可变参数的形式,可以使得函数可以接受1个以上的任意多个参数(不固定)。
注意:
a .可变参数必须从头到尾逐个访问。如果你在访问了几个可变参数之后想半途终止,这是可以的,但是,如果你想一开始就访问参数列表中间的参数,那是不行的。
b.参数列表中至少有一个命名参数。如果连一个命名参数都没有,就无使用va_start。
c. 这些宏是无法直接判断实际存在参数的数量。
d.这些宏无法判断每个参数的是类型。
e.如果在va_arg中指定了错误的类型,那么其后果是不可预测的
这里我们用一个求平均值的栗子说明:
#include <stdio.h>
#include <stdarg.h>
int average(int n, ...)//n个参数
{
int sum = 0; //和
int i = 0;
va_list arg; //创建arg变量,它用于访问参数列表的未确定部分
va_start(arg,n);//初始化过程把 arg 变量设置为指向可变参数部分的第一个参数。
for (i=0;i<n;i++)
{
sum += va_arg(arg,int);//调用时分别取6,9,9
}
return sum/n;
va_end(arg);
}
int main()
{
int ret = average(3, 6,9,9);//3个参数
printf("ret = %d\n",ret);
ret = average(5, 1,2,3,4,5); //5个参数
printf("ret = %d\n",ret);
return 0;
}
输出结果为
这里用到了几个宏,头文件为#include <stdarg.h>
va_start 宏原型为:void va_start(va_list arg_ptr,prev_param ); // (ANSI version)
prev_param 为前一个参数,在此代码中相当于n个参数
va_arg原型为:
type va_arg(va_list arg_ptr,type );//va_arg ,这个宏接受两个参数: va_list 变量和参数列表中下一个参数的类型,va_arg````返回这个参数的值,并使用 va_arg```指向下一个可变参数.
va_end原型为:
void va_end(va_list arg_ptr );
下面再来详细讲解一下这几个宏所代表的什么:
进行宏替换后的代码块:
#include <stdio.h>
#include <stdarg.h>
int average(int n, ...)//n个参数
{
int sum = 0; //和
int i = 0;
//va_list arg; //创建arg变量
char * arg;
//va_start(arg,n);
( arg = (char *)(&n) + 4 );
for (i=0;i<n;i++)
{
//sum += va_arg(arg,int);//调用时分别取6,9,9
sum += ( *(int *)((arg += 4) - 4) );
}
return sum/n;
//va_end(arg);
(arg = (char *)0 );
}
int main()
{
int ret = average(3, 6,9,9);
printf("ret = %d\n",ret);
ret = average(5, 1,2,3,4,5);
printf("ret = %d\n",ret);
return 0;
}
vs 中源码 鼠标点击相应的宏转到定义为:
这里说明一下
_INTSIZEOF(int) //向上取整,当n传的字节大小为5,6,7,8,时宏返回的是8.即1-4时取4,5-8取8
_ADDRESSOF(n) //顾名思义即 &n
各位读者如果有兴趣的话可以自行实现哦!
欢迎大家指教!