最常见的可变参函数就是printf了,它的函数原型为

int printf(const char *format, ...);

  可变参函数至少要有一个确定的参数,后面的...表明该函数第一个参数之后还可以输入若干个参数。

  为解析可变参数,C语言提供了一个va_list类型和,并在stdarg.h中声明了四个函数,分别是va_start, va_arg, va_end, 和va_copy,可以用输入命令:man stdarg 看到它们:

  下面以一个简单的例子讲解:

 1 #include <stdarg.h>
 2 #include <stdio.h>
 3 
 4 void my_printf(const char *fmt, ...)
 5 {
 6     va_list ap;
 7     int num_i;
 8     double num_f;
 9     
10     va_start(ap, fmt);
11     while (*fmt) {
12         if (*fmt == '%') {
13             fmt++;
14             switch (*fmt) {
15             case 'd':
16                 num_i = va_arg(ap, int);
17                 printf("%d", num_i);
18             break;
19 
20             case 'f':
21                 num_f = va_arg(ap, double);
22                 printf("%f", num_f);
23             break;
24             }
25         } else {
26             printf("%c", *fmt);
27         }
28         fmt++;
29     }
30 
31     va_end(ap);
32 }
33 
34 int main(int argc, const char *argv[])
35 {
36     int i = 123, l = 456;
37     double j = 3.14, k = 0.618;
38 
39     my_printf("i = %d\nj = %f\nk = %f\nl = %d\n", i, j, k, l);
40 
41     return 0;
42 }

  第10行:va_start(ap, fmt);  可以理解为 va_start 将参数fmt之后的数据(参数中后面不确定的部分)放入一个临时栈,这个临时栈的指针为ap,指向临时栈的首地址。因为变参占用的空间大小是不确定的,因此必须找到变参中第一个参数,之后便可以根据各个参数的格式,逐一找到后面的参数。取参数的函数为:

type va_arg(va_list ap, type);

  因此在第16行中调用num_i = va_arg(ap, int);表示从临时栈中取出一个int类型的数据,赋值给num_i(同时ap指针后移)同样的,第21行:num_f = va_arg(ap, double); 表示的从栈指针中找出一个double类型的变量赋值个num_f。最后使用va_end(ap)释放临时栈。

 

  再举一个加法的例子:

 1 #include <stdarg.h>
 2 #include <stdio.h>
 3 
 4 /**
 5  * \note cnt表示参与加数的个数,arg1表示第一个加数
 6  */
 7 int add (unsigned int cnt, int arg1, ...)
 8 {
 9     int i, sum = arg1;
10     va_list ap;
11 
12     if (cnt == 0)
13         return 0;
14     if (cnt == 1)
15         return arg1;
16 
17     va_start(ap, arg1);
18 
19     for (i = 1; i < cnt; i++) {
20         sum += va_arg(ap, int);
21     }
22 
23     va_end(ap);
24 
25     return sum;
26 }
27 
28 int main(int argc, const char *argv[])
29 {
30     printf("%d\n", add(3, 10, 20, 30));
31 
32     return 0;
33 }

  add(3, 10, 20, 30)表示对3个数进行相加,分别为10,20,30,那么结果就是60。如果将17行中第二个参数改为cnt(不是已知参数中最后一个参数),结果编译的时候报警告(但是我很纳闷的是为什么输出结果仍然为60):

 值得一提的是,用va_arg取出参数是先输入的参数先取出,因此变参的入栈顺序应该是最后输入的参数最先入栈,因此当可变参函数中变参部分输入的是表达式的时候,应该是先运行后面的表达式,比如下面这段程序:

 1 #include <stdarg.h>
 2 #include <stdio.h>
 3 
 4 /**
 5  * \note cnt表示参与加数的个数,arg1表示第一个加数
 6  */
 7 int add (unsigned int cnt, int arg1, ...)
 8 {
 9     int i, sum = arg1;
10     va_list ap;
11 
12     if (cnt == 0)
13         return 0;
14     if (cnt == 1)
15         return arg1;
16 
17     va_start(ap, arg1);
18 
19     for (i = 1; i < cnt; i++) {
20         sum += va_arg(ap, int);
21     }
22 
23     va_end(ap);
24 
25     return sum;
26 }
27 
28 int main(int argc, const char *argv[])
29 {
30     int a[3] = {10, 20, 30};
31     int *p = &a[1];
32 
33     printf("%d\n", add(3, a[0], *(--p), *(++p)));
34 
35     return 0;
36 }

  运行结果应该是10 + 20 + 30 = 60,而不是10 + 10 + 20。

不仅如此,我们也可以通过下面这段程序推算出定参函数参数的传入顺序(不一定入栈):

 1 #include <stdio.h>
 2 
 3 int add (int arg1, int arg2)
 4 {
 5     return arg1 + arg2;
 6 }
 7 
 8 int main(int argc, const char *argv[])
 9 {
10     int a[3] = {10, 20, 30};
11     int *p = &a[1];
12 
13     printf("%d\n", add(*(--p), *(++p)));
14 
15     return 0;
16 }

输出结果为50而不是30。还可以通过例子说明无论是定参函数还是可变参函数,如果传入的变量是表达式,都是从最后一个参数开始计算,这里就不举例子了。