● 测试代码:
MSDN 上的范例代码,说实话写得很烂。并没有说明变参实现中,几个重要宏的特性。
#include <stdio.h>
#include <stdarg.h> //使用变参,必须引用该头文件。
int Avarage(int nCount,
/*
参数不能完全使用 "
*/
{
va_list params; //定义va_list , 用于访问传入的实参.
va_start(params, nCount); //初始化 va_list, 需要指定为变参前一个参数.
int nTotal =0;
int nBuffer =0;
for(int index =0; index <nCount; index ++)
{
nBuffer =va_arg(params, int); //获取参数,并移动指针到下一位。
nTotal += nBuffer;
}
return nTotal ? (nTotal/nCount):0;
}
int main(int argc, char* argv[])
{
printf("%d",Avarage(2, 10,20));
return 1;
}
运行结果:
15.
● 说明:
C 语言的可变参数实际上是通过
va_list
va_arg
va_end 宏实现的。
其中 va_list 宏可以取得变参中的第一个参数。 (需要指明变参前一个参数)
va_arg 取得变参中的一个参数, 并向下移动指针到下一个参数位置。
由于C 的实现方式, 所以变参函数在实现的时候,都无法确定va_arg 的取值范围。
所以都需要在对变参进行解析之前,明确变参的实际数量。
例如:
printf(char* format, ...);
看起来似乎并没有传入变参的数量, 但实际上 format 中包含的格式字符串已经说明了后面变参的数量,和类型。
如 "%d,%s,%s" 指明一共3个参数, 第一个是整数, 后面两个是字符串。
在Java 中, 变参方法似乎要方便一些。
如: print(String ... args)
在方法里, args 实际是个 String[].
利用 for each 便可以很方便的遍历变参。
或者通过 args.length()获取变参的实际数量。