stdarg.h 可变参数宏的使用

要实现一个可变参数的函数,流程如下:

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");

结果:

stdarg.h 可变参数宏的使用_第1张图片


你可能感兴趣的:(C)