【C】可变参数列表剖析

来源:互联网 发布:部落冲突蛮王升级数据 编辑:程序博客网 时间:2024/04/30 19:36

C 语言中可通过将函数实现为可变参数的形式,使得函数可以接受1个以上的任意多个参数(不固定)。

首先我们来分析一下函数参数,给定下面的一段代码:


我们可以看到 Add()  函数必须给出参数个数,并且实参个数要和定义的形参个数相同,如不同就会发生如上的错误。

再来看一段代码,我们认识一下何为可变参数。

#define _CRT_SECURE_NO_WARNINGS 1#include <stdio.h>int main(){printf("zll\n"); //一个参数printf("%s\n", "zll"); //两个参数printf("%c%s\n", 'z',"ll"); //三个参数system("pause");return 0;


}


运行结果如下:

我们可以发现这三个printf  输出结果相同,但是参数个数明显不同,这就表明了 printf  函数是支持可变参数的函数。我们可以验证一下。

 

从 printf  的定义显然可以证明这一点,方括号中的内容是可有可无的,后边的省略号表明未限制的参数,可自己定义参数。

了解了这些基本的内容后,我们来探讨一下可变参数列表到底是什么,给出代码如下:

#define _CRT_SECURE_NO_WARNINGS 1#include <stdio.h>#include <stdarg.h>int average(int n, ...){va_list arg;int i = 0;int avg = 0;int sum = 0;va_start(arg, n);for (i = 0; i<n; i++){sum += va_arg(arg, int);}avg = sum / n;va_end(arg);return avg;}int main(){int ret1 = average(2, 2, 4);        printf("ret1 = %d\n", ret1);int ret2 = average(5, 1, 2, 3, 4, 5);printf("ret2 = %d\n", ret2);system("pause");return 0;}


运行结果如下:


说明:声明一个 va_list 类型的变量 arg ,它用于访问参数列表的未确定部分。这个变量是调用va_start来初始化的。它的第一个参数是 va_list 的变量名,第2个参数是省略号前最后一个有名字的参数。初始化过程把 arg 变量设置为指向可变参数部分的最后一个确定的参数。

            为了访问参数,需要使用 va_arg ,这个宏接受两个参数: va_list  变量和参数列表中下一个参数的类型。在这个例子中所有的可变参数都是整型。va_arg 返回这个参数的值,并使用 va_arg 指向下一个可变参数。

            当访问完最后一个可变参数之后,我们需要调用 va_end

我们会发现上述的函数就定义了一个可变参数列表,并且实现了我们想要实现的功能,不论我们想要求几个数的平均值都可以,只要我们定义好的大小即可。

可变参数的实现过程是使用宏的封装。只要完成替换,我们就可以分析在不同代码下可变参数的实现过程。





了解了可变参数的实现以后,我们来总结一下可变参数列表的规则和限制:

1)可变参数必须从头到尾逐个访问。如果在访问了几个可变参数之后半途终止是可以的,但是,如果一开始就访问参数列表中间的参数是不行的。

2)参数列表中至少有一个命名参数。如果连一个命名参数都没有,就无法使用 va_start  进行初始化,无法找到第一个未知参数的地址,无法访问参数列表。

3)va_startva_argva_end 这些宏无法直接判断实际存在参数的数量。

4)这些宏无法判断每个参数的类型。

5)如果在 va_arg中指定了错误的类型,那么其后果是不可预测的。


既然已经了解了可变参数列表的定义和限制,那么我们可以编写代码检验一下自己写可变参数列表的能力,代码如下。

#define _CRT_SECURE_NO_WARNINGS 1#include <stdio.h>#include <stdarg.h>int Max(int n, ...){va_list arg;int max = 0;int i = 0;va_start(arg, n);max = va_arg(arg, int);for (i = 0; i < n - 1; i++){int tmp = va_arg(arg, int);if (max < tmp){max = tmp;}}va_end(arg);return max;}int main(){int max = Max(4, 1, 2, 3, 4);printf("max=%d\n", max);system("pause");return 0;}


运行结果如下:


以上就是我关于可变参数列表的理解,如有失误,欢迎指正。

                                             
0 0