您的位置:首页 > 其它

可变参数列表解析

2017-12-10 17:50 134 查看

可变参数

在写c程序时,我们经常使用到printf这个函数,那么在使用这个函数时,我们给他传参吗?传几个参数?

可以看一下函数原型:



可以看出,printf也是有参数的,并且可以传1个以上任意多个参数。

这就是可变参数。

举一个实例来说明:

int average(int n, ...)
{
va_list arg;//定义一个指针变量
int i = 0;
int sum = 0;
va_start(arg, n);//指向可变参数列表部分
for (i = 0; i<n; i++)
{
sum += va_arg(arg, int);//读取每一个参数,并且加到一起
}
return sum / n;
4000

va_end(arg);//结束读取
}
int main()
{

int a = 1;
int b = 2;
int c = 3;
int avg = average(3, a, b, c);
printf("avg = %d\n", avg);

system("pause");
return 0;
}




对于关键宏的解读

此处使用的宏都在stdarg.h头文件中包含。

1.va_list

宏:#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )

实现:实际上只是一个 char * 类型的指针。

2.va_start

宏:#define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) )

这个宏需要两个参数,第一个是上面定义的 va_list, 第二个是可变参数列表之前的那个参数。

注意:va_start中的参数,一定要是已知的最后一个参数,即就是…之前的那个参数。

3.va_arg

宏:#define va_arg(ap,t) ( (t )((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )

同样有两个参数,第一个是前面已经初始化好的 va_list,第二个是类型,比如这里可变参数列表的第一个参数是int类型,那么就传int。

4.va_end

宏:#define va_end(ap) ( ap = (va_list)0 )

为了更好的可读性,结束可变参数的获取。

可变参数的限制

注意

可变参数必须从头到尾访问。可以在访问几个可变参数之后终止,但不能从一开始就访问参数列表中间的参数。

参数列表中至少要有一个已知的命名参数,不能一个都没有,否则无法使用va_start。

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

并且宏也无法判断每个参数的类型。

实现printf函数

再了解了可变参数列表之后,可以实现一个我们自己的printf函数。

#include<stdio.h>
#include<stdlib.h>
#include<stdarg.h>
void print(char *format,...)
{
va_list(arg);
va_start(arg, format);
while (*format)
{

switch (*format)
{
case 's':
{
char *str = va_arg(arg, char*);
while (*str)
{
putchar(*str);
str++;
}
}
break;
case 'c':
{
char c = va_arg(arg, char);
putchar(c);
}
break;
default:
break;

}
format++;
}
va_end(arg);
}
int main()
{
print("s cccccc d.\n","hello",'\t','w','o','r','l','d',100)

system("pause");
return 0;
}


内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: