前言
??在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_start(va_list va, 参数);
2、取可变参数
va_arg(va_list va, int);
3、结束
va_end(va_list va);
2.1 取自C和指针的例子
#include <stdio.h>
#include <stdarg.h>
float average(int val_num,...)
{
va_list va;
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()函数,虽然也是可变参数列表,但其格式化字符串中指定了参数的数量和类型.
解析格式化字符串中的参数数量和类型也会带来一些花销,且有时传入格式化字符串是不必要的。此时就得保证可变参数列表的参数类型和数量的合法性,比如使用数组传入参数。
|