像printf和scanf这些函数的参数都是可变的,那么如何定义可变参数呢?
这要解决两个问题
1、如何声明
2、如何调用
[b]如何声明:[/b]
printf的声明形式如下:
就这么简单。这里有一点要注意一下,就是至少要有一个有名参数,否则不能定义可变参数,如果定义为printf(...),编译时会报错“ISO C requires a named argument before '...'”
[b]如何调用:[/b]
这个比较麻烦,名字都没有怎么调用呢。
<stdarg.h>中定义了一组宏和类型,用来帮助我们遍历参数列表
va_list类型用来声明一个变量
va_start(v,l)这个宏用结果是将v指向第一个无名参数,v是我们声明的va_list类型,l是最后一个有名参数。
va_arg(v,l)返回一个参数,并将v指向下一个参数,这里的l与上面不同,是类型的名字,比如int,double等等
va_end(v,l)与start对应,在函数返回前调用这个,做一些清理工作
下面举个例子,我们模仿printf写一个简单的版本:
关键部分都有注释了,一看就明白
这要解决两个问题
1、如何声明
2、如何调用
[b]如何声明:[/b]
printf的声明形式如下:
int printf(char *fmt,...)
就这么简单。这里有一点要注意一下,就是至少要有一个有名参数,否则不能定义可变参数,如果定义为printf(...),编译时会报错“ISO C requires a named argument before '...'”
[b]如何调用:[/b]
这个比较麻烦,名字都没有怎么调用呢。
<stdarg.h>中定义了一组宏和类型,用来帮助我们遍历参数列表
va_list类型用来声明一个变量
va_start(v,l)这个宏用结果是将v指向第一个无名参数,v是我们声明的va_list类型,l是最后一个有名参数。
va_arg(v,l)返回一个参数,并将v指向下一个参数,这里的l与上面不同,是类型的名字,比如int,double等等
va_end(v,l)与start对应,在函数返回前调用这个,做一些清理工作
下面举个例子,我们模仿printf写一个简单的版本:
#include <stdarg.h>
void myprintf(char *s,...) //注意这个声明
{
va_list argp; //声明一个arg类型
char *p,*svalue;
int ivalue;
double dvalue;
va_start(argp,s); //argp指向第一个无名参数
for(p=s;*p;p++){
if(*p!='%'){
putchar(*p);
continue;
}
switch (*++p){
case 'd':
ivalue = va_arg(argp,int);//获取一个int型的变量值,argp指向下一个变量
// 略....输出整型值
break;
case 'f':
dvalue = va_arg(argp,double);//获取一个double型的变量值,argp指向下一个变量
// 略....输出double型值
break;
case 's':
svalue = va_arg(argp,char *);//获取一个字符串型的变量值,argp指向下一个变量
// 略....输出字符串
break;
default:
putchar(*p);
break;
}
}
va_end(argp);//做清理工作
}
关键部分都有注释了,一看就明白