建立参数个数不定的函数是可能的。如标准库函数printf,它的参数个数是不
定的。虽然printf至少必须用一个字符串作为其第一个参数,但是它能够接收任何个数
的附加参数。printf函数的原型为:
int printf(const char *format, ...);
该函数原型中的省略号(...)表示这个函数接收个数不定的任何类型的参数。注意省略号
必须放在参数列表的最后。
变长参数头文件stdarg.h中的宏定义(见下表)课用来建立变长参数列表。下面给
出的演示程序中给出了接收可变参数个数的函数average。函数average的第一个参数是要
被求平均值的数据的个数。
+-----------------------------------------------------------------------------+
| 在头文件stdarg.h中定义的类型和宏 |
+-----------------------------------------------------------------------------+
| 标识苻 | 解释 |
+----------+------------------------------------------------------------------+
| va_list | 用来保存va_start, va_end所需信息的一种类型。为了访问变长参数列 |
| | 表中的参数,必须声明va_list类型的一个对象 |
+----------+------------------------------------------------------------------+
| va_start | 访问变长参数列表中的参数之前使用的宏,它初始化用va_list声明的 |
| | 的对象,初始化结果供宏va_arg 和va_end使用。 |
+----------+------------------------------------------------------------------+
| va_arg | 展开成一个表达式的宏,该表达式具有变长参数列表中下一个参数的 |
| | 值和类型。每次调用va_arg都会修改用va_list声明的对象,从而使该 |
| | 对象指向参数列表中的下一个参数。 |
+----------+------------------------------------------------------------------+
| va_end | 该宏使程序能够从变长参数列表用宏va_start应用的函数中正常返回 |
+----------+------------------------------------------------------------------+
函数average使用了头文件stdarg.h中所有的年个月和宏。宏va_start, va_arg
和va_end 用va_list类型的对象ap处理函数avergage的变长参数列表。 函数先用
va_start初始化供va_arg 和va_end 使用的对象ap。 宏va_start有两个参数,一个是
对象ap,另各一个是变长参数列表中省略号前的最右边参数的标识符(本例中为i,
va_start用i确定变长参数列表的起始位置)。 然后,函数average反复把变长参数
列表中的参数与变量total相加,加到total中值是用宏va_arg从变长参数列表中检索
到的。宏va_arg有两个参数,一个是对象ap,另一个是要从参数列表中接收的值的类型
(本例中为double),它返回该参数的值。宏va_end只有对象ap这样一个参数。函数
average用宏va_end使得程序从average正常返回到main。最终,函数计算平均值返回到
main。
常见的程设计错误是把省略号放在函数参数列表中间。省略号只能放在参数
列表的最后。那么函数printf和scanf是怎样知道宏va_arg每次使用的类型的呢?答案
是printf和scanf是通过扫描格式控制串中的格式转换苻来确定所要处理的下一个参数
的类型的。
附测试程序:VarParam.c, RedHat Linux 9.0 下测试通过。
#include
#include
double average(int, ...);
main()
{
double w = 37.5, x = 22.5, y = 1.7, z = 10.2;
printf("w = %.1f\nx = %.1f\ny = %.1f\nz = %.1f\n", w, x, y, z);
printf("the average of w and x is : %f.1\n", average(2, w, x));
printf("the average of w, x and y is : %f.1\n", average(3, w, x, y));
printf("the average of w, x, y and z is : %f.1\n", average(4, w, x, y, z));
}
double average(int i, ...)
{
double total = 0;
int j;
va_list ap;
va_start(ap, i);
for (j = 1; j