可变参数列表分析
C语言中的可变参数是一个比较有意思的实现,通过将函数实现为可变参数的形式,可以使得函数接受1个以上的任意多的参数(不固定)。
例:使用可变参数,实现函数,求多个函数参数的平均值。
#include<stdio.h>
#include<stdarg.h>
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;
va_end(arg);
}
int main()
{
int a = 1;
int b = 2;
int c = 3;
int avg = average(3, a, b, c);
printf("avge = %d\n", avg);
system("pause");
return 0;
}
- 声明一个 va_list 类型的变量 arg,它用于访问参数列表的未确定部分。
- 这个变量是调用 va_start 来初始化的,它的第1个参数是 va_list 的变量名,第2个参数是省略号前最后一个有名字的参数,初始化过程把 arg 变量设置为指向可变参数部分的第一个参数。
- 为了访问参数,需要使用 va_arg ,这个宏接受两个参数:va_list 变量和参数列表中下一个参数的类型,在这个例子中所有的可变参数都是整形,va_arg 返回这个参数的值,并用 va_arg 指向下一个可变参数。
- 最后,当访问完最后一个可变参数之后,我们需要调用 va_end。
可变参数限制
- 可变参数必须从头到尾逐个访问,如果你在访问了几个可变参数之后想半途终止,这是可以的,但是,如果你想一开始就访问参数列表中间的参数那是不行的。
- 参数列表中至少有一个命名参数,如果连一个命名参数都没有就无法使用 va_start。
- 这些宏是无法直接判断实际存在参数的数量。
- 这些宏无法判断每个参数的类型。
- 如果在 va_arg 中指定了错误的类型,那么其后果是不可预测的。
vs中源码解析
可变参数是宏的封装,只要完成替换就可自行分析了。
小练习:
//1.使用可变参数,实现函数,求函数参数的最大值。
#include<stdio.h>
#include<stdarg.h>
int Max(int n, ...)
{
va_list arg;
int i = 0;
int max = 0;
va_start(arg, n);
for (i = 0; i < n; i++)
{
int val = va_arg(arg, int);
if (val > max)
{
max = val;
}
}
return max;
va_end(arg);
}
int main()
{
int a = 10;
int b = 30;
int c = 20;
int ret = Max(3, a, b, c);
printf("max = %d\n", ret);
system("pause");
return 0;
}
2..模拟实现printf函数,可完成下面的功能能完成下面函数的调用。
//print(“s ccc d.\n”,”hello”,’b’,’o’,’y’,100);
//函数原型: print(char *format, …)
#include<stdio.h>
#include<stdarg.h>
void Decimal(int num)
{
if (num / 10 == 0)
putchar(num % 10 + '0');
else {
Decimal(num / 10);
putchar(num % 10 + '0');
}
}
void print(const char *fmt, ...)
{
int i = 0;
va_list arg;
va_start(arg, fmt);
for (i = 0; fmt[i] != '\0'; i++) {
if (fmt[i] != '%')
putchar(fmt[i]);
else {
i++;
switch (fmt[i]) {
case 'c':
putchar(va_arg(arg, int)); break;
case 'd': {
int r = va_arg(arg, int); Decimal(r); }break;
case 's':{
char *r = va_arg(arg, char*);
for (int i = 0; r[i] != '\0'; i++) putchar(r[i]);
}
break;
case '%':
putchar('%');
break;
}
}
}
}
int main()
{
print("s ccc d.\n", "hello", 'b', 'o', 'y', 100);
system("pause");
return 0;
}