可变参数列表码的剖析和自己的理解

在之前学习的所有函数中,我们所经行参数的传递,形参实参的个数都是有限的。但是c语言的库函数printf()的参数个数却是不确定。我们称其为可变参数列表。


之后我们经行可变参数列表的模拟实现.在模拟实现过程中,需要用到几个宏:

#include<stdarg.h>
va_list arg;\\声明arg
va_start(arg,format);\\初始化arg
va_arg(arg,type);\\向后移动类型所占字节
va_end(arg);\\结束

举例实现可变参数列表的实现;设计一个函数求参数的平均值

一般实现:

#include<stdio.h>
int average(int n, ...)
{
	int num = 0;
	char*p = &n;
	int i = 0;
	for (i = 1; i <=n; i++)
	{
		num = num+(*(p +(4*i)));
	}
	return num / n;
}
int main()
{
	int a = 2;
	int b = 4;
		int c = 6;
		int arg1 = average(3, a, b, c);
		int arg2 = average(2, a, b);
		printf("%d\n", arg1);
		printf("%d\n", arg2);
}

由函数栈桢的知识可知,创建栈桢时函数的参数从左向右依次入栈。所以,当我们取到第一个参数地址,依次加类型所占字节数,便可以得到随后参数的地址。解引用得到参数的值经行运算。


宏实现:

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<stdarg.h>
int average(int n, ...)
{
	va_list arg;
	int i = 0;
	int num = 0;
	va_start(arg, n);
	for (; i < n; i++)
	{
		num = num + va_arg(arg, int);
	}
	return num / n; 
	va_end(arg);
}

int main()
{
	int a = 0;
	int b = 0;
	int c = 0;
	int n = 0;
	scanf("%d %d %d %d", &n,&a,&b,&c);
	int arg = average(n, a, b, c);
	printf("%d", arg);
}


声明一个 va_list 类型的变量 arg ,它用于访问参数列表的未确定部分。这个变量是调用 va_start 来初始化的。它的第一个参数是 va_list 的变量名,第2个参数是省略号前最后一个有名字的参数。初始化过程把 arg 变量设置为指向可变参数部分的第一个参数。为了访问参数,需要使用 va_arg ,这个宏接受两个参数: va_list 变量和参数列表中下一个参数的类型。在这个例子中所有的可变参数都是整型。va_arg````返回这个参数的值,并使用 va_arg```指向下一个可变参数。最后,当访问完毕最后一个可变参数之后,我们需要调用 va_end 。


va_list arg相当于char *p声明arg;va_start(arg,n)对arg初始化使其指向n后位置;va_arg(arg,int)使用一次增加类型所占字节个数的地址,第一次指向n后;va_end(arg)释放arg。

可变参数列表模拟完成。


注意:

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

2.参数列表中至少有一个命名参数。如果连一个命名参数都没有,就无法使用va_start。

3.这些宏是无法直接判断实际存在参数的数量。这些宏无法判断每个参数的是类型。

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





猜你喜欢

转载自blog.csdn.net/wakelee14122/article/details/78772569