函数的参数一定是固定的吗?
答案是否定的!
有一个东西叫可变参数
#define _CRT_SECURE_NO_WARNINGS 1
//可变参数 求平均值
#include<stdio.h>
#include<stdarg.h>
int avg(int a, ...)
{
va_list arg;
int i = 0;
int sum = 0;
va_start(arg, a);
for(i=0; i<a; i++)
{
sum = sum + va_arg(arg, int);
}
return sum/a;
va_end(arg);
}
int main()
{
int a = 1;
int b = 2;
int c = 3;
int avg1 = avg(2, a, c);
int avg2 = avg(3, a, b, c);
printf("avg1 = %d\n", avg1);
printf("avg2 = %d\n", avg2);
return 0;
}
有了可变参数, 我们可以求任意个数的平均值。
使用步骤
01>在函数里定义一个va_list型的变量,这里是arg,这个变量是指向参数的指针
02>用va_start宏初始化变量arg,这个宏的第二个参数是第一个可变参数的前一个参数,是一个固定的参数
03>用va_arg返回可变的参数,va_arg的第二个参数是你要返回的参数的类型,这里是int型
04>用va_end宏结束可变参数的获取,然后你就可以在函数里使用第二个参数了,如果函数有多个可变参数的,依次调用va_arg获取各个参数
注意!
01.可变参数必须从头到尾逐个访问。如果你在访问了几个可变参数之后想半途终止,这是可以的,但是,如果你想一开始就访问参数列表中间的参数,那是不行的
02.参数列表中至少有一个命名参数。如果连一个命名参数都没有,就无法使用va_start
03.这些宏无法直接判断实际存在参数的数量
这些宏无法判断每个参数的类型
如果在va_arg中指定了错误的类型,那么其后果是不可预测的
我们最常用的printf函数 可以发现其实它的参数就是可变参数
我们就来自己实现一下打印函数
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdarg.h>
void print(char *format, ...)
{
char *ch;
va_list arg;//定义arg
va_start(arg, format);//初始化arg 把可变参数列表中首个变量地址给arg
while (*format)
{
int d = 0;
int ret = 0;
switch(*format)
{
case 'c':
putchar(va_arg(arg,char));//va_arg 获取参数列表中下一个未知参数,可以通过前面获取的字符判断下一个参数类型
break;
case 's':
ch = va_arg(arg, char*);//定义一个指针变量接收获取的字符,用putchar()一个一个输出
while (*ch)
{
putchar(*ch);
ch++;
}
break;
case 'd':
d = va_arg(arg, int);
printf("%d", d);
while(d)
{
ret++;
d /= 10;
}
break;
default :
putchar(*format);
}
format++;
}
va_end(arg);//释放arg
}
int main()
{
print("s ccc d\n","hello", 'b', 'i', 't', 100);
system("pause");
return 0;
}