天天看点

C语言可变参数列表解析

我们通过下面这个简单的例子来了解一下C语言可变参数列表的使用:

#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>

int get_max(int n,...)
{
	va_list arg;                        //声明一个va_list变量arg,用于访问参数列表未确定的部分
	va_start(arg, n);                   //第一个参数是va_list变量,第二个参数是省略号前最后一个有名字的参数
	int i = 0;                          //初始化过程把arg变量设置为指向可变参数部分的第一个参数
	int temp = 0;
	int max = va_arg(arg,int);          //访问可变参数,传入va_arg的第一个参数是va_list变量,第二个是参数列表中下一个参数的类型
	for (i = 0; i < n-1; i++)
	{
		temp = va_arg(arg,int);
		if (max < temp)
		{
			max = temp;
		}
	}
	return max;
        va_end(arg);                        //访问完最后一个参数后,调用va_end
 }

int main()
{
	int a = 1;
	int b = 4;
	int c = 3;
	printf("%d\n",get_max(3,a,b,c));
	system("pause");
	return 0;
}
           

一些需要注意的点:

1、可变参数必须从头至尾逐个访问,不可直接访问中间的参数。

2、参数列表中至少有一个命名参数。否则无法使用va_start。

3、这些宏无法判断参数的数量和类型。

4、如果va_arg中指定了错误的类型,结果无法预测。

我们来看一下可变参数列表的源码:

#ifdef  _M_ALPHA
typedef struct {
        char *a0;       /* pointer to first homed integer argument */
        int offset;     /* byte offset of next parameter */
} va_list;
#else
typedef char *  va_list;  //va_list其实就是char *
#endif
#define _VA_LIST_DEFINED
#endif 


#ifdef  _M_IX86


#define _INTSIZEOF(n)   ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1)) 
#define va_start(ap,v)  ( ap = (va_list)&v + _INTSIZEOF(v) )          
#define va_arg(ap,t)    ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
#define va_end(ap)      ( ap = (va_list)0 )