函数printf的正确申明形式为
int printf(char *format,...);
其中,省略号表示参数表中参数的数量和类型都是可变的,省略号只能出现在参数表的末尾,因为miniprintf函数不需要像printf函数一样返回实际输出的字符数,因此我们将它申明为一下形式
void miniprintf(char *format,...);
编写函数miniprintf的关键在于如何处理出来一个甚至连名字都没有的参数表,标准头文件<stdarg.h>中包含了一组宏定义,他们队如何遍历参数表进行了定义,这个头文件的实现因为不同的机器有着不同的实现,但是提供的接口是一致的。
va_list类型用于声明一个变量,该变量一次引用各参数,在函数miniprintf中,我们将该变量称为ap,意思是“参数指针”,宏va_start将ap初始化为指向第一个无名参数的指针,在使用ap之前,该宏必须被调用一次,参数表必须至少包括一个有名参数,va_start将最后一个有名参数作为起点。
每次调用va_arg,该函数返回一个参数,并将ap指向下一个参数,va_arg使用一个类型名来决定返回的对象类型,指针移动的步长,最后必须在函数返回之前调用va_end,以完成一些必要的清理工作
我们简化的miniprinf函数如下图所示
#include <stdio.h>
#include <stdarg.h>
void miniprintf(char *format, ...) {
va_list ap;
char *p, *sval;
int ival;
double dval;
va_start(ap, format);
for (p = format; *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;
default:
putchar(*p);
break;
}
}
va_end(ap);
}
int main() {
char *s = "123";
int d = 123;
double f = 123.123;
miniprintf("%s-%d-%f", s, d, f);
return 0;
}