C语言中的printf函数的完整声明为:int printf(char *fmt, ...);
其中,省略号表示参数表中参数的数量和类型是可变的,并且省略号只能出现在参数表的尾部。这里编写一个与printf函数功能类似的函数miniprintf来演示变长参数表的实现,声明如下:void miniprintf(char *fmt, ...);
标准头文件<stdarg.h>中包含了一组宏定义,它们对如何遍历参数表进行了定义:
- va_list类型用于声明一个变量,该变量将依次引用各参数,程序中即ap
- 宏va_start将ap初始化为指向第一个无名参数的指针。在使用ap之前,该宏必须被调用一次。参数表必须至少包括一个有名参数,va_start将最后一个有名参数作为起点
- 每次调用va_arg,该函数返回一个参数,并将ap指向下一个参数。va_arg使用一个类型名来决定返回的对象类型、指针的移动步长。
- va_end在函数返回之前调用,做一些清理工作
以下是miniprintf函数的实现及测试(参考《The C Programming Language》):
void miniprintf(char *fmt, ...);
int main()
{
miniprintf("I will print %d %s floats: %f %f %f\n", 3, "strange", 0.1, 0.998, 0.235);
return 0;
}
void miniprintf(char *fmt, ...)
{
va_list ap;
char *p, *sval;
int ival;
double dval;
va_start(ap, fmt);
for (p = fmt; *p; p++) {
if (*p != '%') {
putchar(*p);
continue;
}
switch (*++p) {
case 'd':
ival = va_arg(ap, int);
printf("%d", ival);
break;
case 'f':
dval = va_arg(ap, double);
printf("%f", dval);
break;
case 's':
for (sval = va_arg(ap, char *); *sval; sval++)
putchar(*sval);
break;
case '\n':
printf("\n");
break;
default:
putchar(*p);
break;
}
}
va_end(ap);
}