【C语言】可变参数

来源:互联网 发布:labview串口数据采集 编辑:程序博客网 时间:2024/06/06 01:27

在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;}
原创粉丝点击