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_arg ,这个宏接受两个参数: va_list 变量和参数列表中下一个参数的类型。在这个例子中所有的可变参数都是整型。va_arg 返回这个参数的值,并使用 va_arg 指向下一个可变参数。
当访问完最后一个可变参数之后,我们需要调用 va_end 。
我们会发现上述的函数就定义了一个可变参数列表,并且实现了我们想要实现的功能,不论我们想要求几个数的平均值都可以,只要我们定义好 n 的大小即可。
可变参数的实现过程是使用宏的封装。只要完成替换,我们就可以分析在不同代码下可变参数的实现过程。
了解了可变参数的实现以后,我们来总结一下可变参数列表的规则和限制:
1)可变参数必须从头到尾逐个访问。如果在访问了几个可变参数之后半途终止是可以的,但是,如果一开始就访问参数列表中间的参数是不行的。
2)参数列表中至少有一个命名参数。如果连一个命名参数都没有,就无法使用 va_start 进行初始化,无法找到第一个未知参数的地址,无法访问参数列表。
3)va_start、va_arg、va_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;
}
以上就是我关于可变参数列表的理解,如有失误,欢迎指正。