前言
在C语言中说到可变参数,其实大家最熟悉的printf()函数就是可变参数。因为我们不仅可以单单打印字符串:
printf("Helloworld!\n");
还可以使用他来输出数值:
int num = 0;
printf("Hello, %d\n", num);
1. 可变参数举例
使用printf
打印format
.
#define LOG(format, ...)\
{\
printf("%s, %d" format, __FUNCTION__, __LINE__, ##__VA_ARGS__);\
}
其实 `__VA_ARGS__` 就是一个可变参数的宏,替代上面的...
`##__VA_ARGS__` 就是当可变参数个数为0时,将参数列表中前面多余的,去掉.
在这个例子中,在" "
后,printf仍可以接受format参数。__FUNCTION__和__LINE__
的值来表示%s和%d,在后面是真正要打印到format
的参数,或者简化一下更清晰:
#define LOG(format, ...)\
{\
printf(format, ##__VA_ARGS__);\
}
但所有内容都得传进format
:
LOG("Func:%s Line:%d ProblemLevel: ", __FUNCTION__, __LINE__);
2. 可变参数实现原理
可变参数列表是通过宏来实现的,定义在stdarg.h中。
#include <stdarg.h>
//一个类型
valist;
//三个宏函数:
va_start()
va_arg()
va_end()
1、初始化
/*
功能:初始化过程把 va_list 类型的变量 value 指向可变参数部分的第1个参数
参数1:va_list类型变量
参数2:省略号前最后一个有名字的参数
返回值:无
*/
va_start(va_list va, 参数);
2、取可变参数
/*
功能:返回可变参数,并使得va指向下一个可变参数
入参1:va_list类型变量
入参2:参数列表中下一个参数的类型
返回值:
*/
va_arg(va_list va, int);
3、结束
va_end(va_list va);
2.1 取自C和指针的例子
/*求n个整形参数的平均值,利用可变参数列表来实现
形参列表中必须要有一个固定值,不能只有省略号所代表的未确定参数*/
#include <stdio.h>
#include <stdarg.h>
float average(int val_num,...)
{
va_list va; //va_list类型声明
float sum=0;
/*准备访问可变参数*/
va_start(va, val_num); //初始化参数,指向第一个可变参数
/*添加取自可变参数列表的值*/
for(int count=0; count<val_num; count++)
{
sum += va_arg(va, int); //取当前参数,并指向下一个参数
}
va_end(va); //完成可变参数的处理
float average=((float)((int)((sum/values+0.005)*100)))/100;
return average;
}
#define val_num 3
int main()
{
float aver;
aver=average(val_num, 9, 9, 8);
return 0;
}
2.2 使用限制
从上面的例子上上可以看出使用va_list
类型及其三个宏函数的限制:
- 参数列表至少要有一个命名参数(确定类型的参数).(这一般都能满足,似乎不是什么问题.)
- 这些宏无法判断实际存在的参数的数量.
- 这些宏也无法判断每个参数的类型.
要解决上面两个问题,办法就是使用命名参数。
例如printf()函数,虽然也是可变参数列表,但其格式化字符串中指定了参数的数量和类型.
解析格式化字符串中的参数数量和类型也会带来一些花销,且有时传入格式化字符串是不必要的。此时就得保证可变参数列表的参数类型和数量的合法性,比如使用数组传入参数。