我最近对可变参数列表十分感兴趣,因为它可以接受不同数目的参数。让我们先看一下它的声明:
void printf (const char *format,...)
我们通过一个具体的例子来理解它:
#include <stdio.h> #include <stdarg.h> int average (int val, ...) { int i = 0; int sum = 0; va_list arg;//定义了一个char 类型的指针 va_start(arg,val);//读取所有的参数,从头开始,后读取你所给的参数个数,并且把参数放到指针中 for (i = 0; i < val; i++) { sum += va_arg(arg,int); } va_end(arg);//结束读取 return sum / val; } int main() { int ret = 0; ret = average(5,1,2,3,4,5); printf("%d\n",ret); return 0; }
可变参数列表是通过宏-stdarg来实现的,这个头文件中声明了一个类型va_list和三个宏va_start,va_arg,va_end,通过右击并转到定义处,我们可以发现va_list其实就是一个char *的重命名,函数中是这样写的,typedef char * va_list,这就非常好理解了;对于va_start,va_arg和va_end三个宏我们分别转到定义处可以发现va_start其实是这样的:,而_crt_va_start依然是由define定义的标识符,其实应该是这样( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) );而va_arg和va_start类似也是由define定义的标识符,它的原型应该是这样的:#define _crt_va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) ),ap就是指向我的可变参数列表中的第一个参数,通过它可以找到之后的参数;对于va_end它的原型应该是这样的#define _crt_va_end(ap) ( ap = (va_list)0 ),用于结束va_list说指向的参数列表,当访问完最后一个可变参数之后,我们就需要调用va_end来结束指向的参数列表