要实现一个可变参数的函数,流程如下:
1.定义一个函数,该函数有固定参数,且最后一个固定参数后面是可变参数(用“...”表示)
2.在该函数体内定义va_list类型的变量
3.用va_start宏将该变量初始化为一个参数列表,其指向可变参数列表的第一个元素(即在使用时,匹配固定参数后面的第一个实参)
4.将此变量作为va_arg宏或vsnprintf等函数的参数,访问这个可变参数列表里面的元素
5.用va_end宏完成清理工作
写个可变参数求取总和的函数:
int retsum0(int a, ...)
{
int sum=0;
va_list ap;
va_start(ap,a);
for(int i=0;i
int retsum1(int a, ...)
{
int sum=0,tmp;
va_list ap;
va_start(ap,a);
while((tmp=va_arg(ap,int))!=-1)
sum+=tmp;
va_end(ap);
return sum;
}
两种方法,取决于如何调用该函数。尽管每次调用va_arg都返回下一个实参(可变参数列表里面的),但是它并不知道用户到底传了几个实参,盲目访问只会导致越界。
因此只能通过人为方法判断,通常最后一个固定参数建议设定为可变参数数目;或者最后一个可变实参设定为一个特殊值。通过判断固定参数、最后的可变实参来确定可变参数何时访问完了。
调用:
printf("sum is %d\n",retsum0(4,20,30,40,2));
printf("sum is %d\n",retsum1(9999,20,30,40,2,-1));
结果:
然后写个可变参数的字符串(可按格式)打印函数:
void printvar0(const char* args,...)
{
char buf[MAXLINE]={0};
va_list ap;
va_start(ap,args);
while(args!=0)
{
strcat(buf,args);
printf("cur buf: %s\n",buf);
args = va_arg(ap, const char *);
printf("cur a: %p ===>%s\n",args,args);
}
strcat(buf,"\n");
fputs(buf,stdout);
fflush(NULL);
va_end(ap);
}
void printvar1(const char* a,...)
{
char buf[MAXLINE]={0};
va_list ap;
va_start(ap,a);
vsnprintf(buf,MAXLINE,a,ap);
fputs(buf,stdout);
fflush(NULL);
va_end(ap);
}
这里有直接输出和按格式输出两个版本。前者判断args!=0正是Linux execl API的做法;后者的vsnprintf是printf函数族的一员,它接受的最后两个参数分别是格式化字符串和va_list变量。
调用:
printvar0("AAA","BBB","CCC","DDD",(char*)0);
printvar1("%s<=======\n","BB");
结果: