【C】可变参数列表剖析

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;
}


运行结果如下:


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





评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值