在C语言中学习中,我经常用到的printf、scanf就是一个参数个数可变的函数。
int printf( const char* , ...);
int scanf(const char *, ...);
介绍C语言中有关可变参数的宏
void va_start( va_list arg_ptr, prev_param );
type va_arg( va_list arg_ptr, type );
void va_end( va_list arg_ptr )
va_list:用来保存宏va_start、va_arg和va_end所需信息的一种类型。为了访问变长参数列表中的参数,必须声明va_list类型的一个对象。
转到定义,可以看到: typedef char * va_list;
va_list实际上使一个 char*类型
va_start:访问变长参数列表中的参数之前使用的宏,它初始化用 va_list声明的对象,初始化结果供宏va_arg和 va_end使用;
转到定义,查看va_start的具体实现:
#define _INTSIZEOF(n) \
( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
#define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) )
其中宏_INTSIZEOF(n)的作用就是把n提升到4的倍数 ,如果n为1,2,3,4 那么 该值为4,如果为5,6,7,8 那么该值为8……..
现在看va_start :该宏使ap指向下一个可变参数。
注意:va_start第二个参数v,一定是函数参数最后一个有名的参数,或者说是可变参数列表的前一个参数。
va_arg: 展开成一个表达式的宏,该表达式具有变长参数列表中下一个参数的值和类型。每次调用va_arg都会修改用va_list声明的对象,从而使该对象指向参数列表中的下一个参数;
查看定义:
#define va_arg(ap,t)
( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
这个宏写的很巧妙:先看后面的一个括号里面的内容
(ap += _INTSIZEOF(t)),使ap指向下个可变参数,
然后整体宏的结果是当前可变参数的结果。
va_end:该宏使程序能够从变长参数列表用宏va_start引用的函数中正常返回。
#define va_end(ap) ( ap = (va_list)0 )
使用可变参数的步骤
1)首先在函数里定义一个va_list型的变量,这里是arg,这个变量是指向参数的指针.
2)然后用va_start宏初始化变量arg,这个宏的第二个参数是第一个可变参数的前一个参数,是一个固定的参数.
3)然后用va_arg返回可变的参数, va_arg的第二个参数是你要返回的参数的类型,这里是int型. 如果函数有多个可变参数的,依次调用va_arg获取各个参数.
4)最后用va_end宏结束可变参数的获取.
实现一个求平均值函数
#include <stdio.h>
#include <stdarg.h>
int average(int val,...)
{
int sum = 0;
int i = 0;
va_list arg;
va_start(arg,val);//val必须为可变参数的前一个参数(最后一个有名参数)
for(i=0; i < val; ++i)
{
sum+=va_arg(arg,int);
}
va_end(arg);
return sum/val;
}
int main()
{
int ret = average(2,10,20);
printf("%d\n",ret);
return 0;
}