可变参数列表是通过宏来实现的,这些宏定义于stdarg.h头文件中,它是标准库的一部分。
这个头文件声明了一个类型va_list和三个宏:va_start/va_arg/va_end.我们可以声明一个类型为va_list的变量,与这几个宏配合使用,来访问参数的值。
/*
功能:求n_values个int类型数的和。
*/
float average(int n_values,...)
{
va_list var_arg;
int count;
float sum = 0.0;
va_start(var_arg, n_values);
for (count =0;count <n_values; count++)
{
sum += va_arg(var_arg,int);
}
va_end(var_arg);
return sum /n_values;
}
1> va_start初始化var_arg变量,第2个参数是省略号前最后一个有名参数。
2> 初始化过程把var_arg变量设置为指向可变参数部分的第1个参数。
3> 为了访问参数,使用va_arg,这个宏接受两个参数:va_list变量和参数列表中下一个参数的类型。
4> 访问最后一个可变参数之后,调用va_end
NOTE:由于参数列表中的可变参数部分没有原型,所有作为可变参数传递给函数的值都将执行缺省参数类型提升。
char/short/和float类型的值实际上将作为int或者double类型的值传递。
/*
支持%s,%d,%x,%c,%f
my_printf("my name is :%s, my age :%d, my char:%c, my float %f", "Hello",16, 'c', 1.23);
*/
int my_printf(char* fmt,...)
{
va_list var_arg;
va_start(var_arg,fmt);
char *str = NULL;
int val;
char c;
float f;
while(*fmt)
{
while(*fmt != '%')
{
putchar(*fmt);
fmt ++;
if (*fmt == '\0')
{
return 0;
}
}
fmt ++;
switch(*fmt)
{
case 's':
str = va_arg(var_arg,char*);
printf("%s",str);
break;
case 'd':
val = va_arg(var_arg,int);
printf("%d",val);
break;
case 'x':
val = va_arg(var_arg,int);
printf("%x",val);
break;
case 'c':
c = va_arg(var_arg, int); //类型提升
printf("%c",c);
break;
case 'f':
f = va_arg(var_arg,double); //类型提升
printf("%f",f);
break;
default:
printf("error %c param\n", *fmt);
}
fmt++;
}
va_end(var_arg);
return 0;
}