可变参数列表解析

以下均在vs2008 中实现

通过将函数实现为可变参数的形式,可以使得函数可以接受1个以上的任意多个参数(不固定)。

注意:
a .可变参数必须从头到尾逐个访问。如果你在访问了几个可变参数之后想半途终止,这是可以的,但是,如果你想一开始就访问参数列表中间的参数,那是不行的。
b.参数列表中至少有一个命名参数。如果连一个命名参数都没有,就无使用va_start。
c. 这些宏是无法直接判断实际存在参数的数量。
d.这些宏无法判断每个参数的是类型。
e.如果在va_arg中指定了错误的类型,那么其后果是不可预测的

这里我们用一个求平均值的栗子说明:

#include <stdio.h>
#include <stdarg.h>
int average(int n, ...)//n个参数
{
    int sum = 0;  //和
    int i = 0;
    va_list arg;  //创建arg变量,它用于访问参数列表的未确定部分
    va_start(arg,n);//初始化过程把 arg 变量设置为指向可变参数部分的第一个参数。
    for (i=0;i<n;i++)
    {
        sum += va_arg(arg,int);//调用时分别取6,9,9
    }
    return sum/n;
    va_end(arg);
}
int main()
{
    int ret = average(3, 6,9,9);//3个参数
    printf("ret = %d\n",ret);

    ret = average(5, 1,2,3,4,5); //5个参数
    printf("ret = %d\n",ret);


    return 0;
}

输出结果为
这里写图片描述

这里用到了几个宏,头文件为#include  <stdarg.h>
va_start 宏原型为:void va_start(va_list arg_ptr,prev_param ); // (ANSI version)
prev_param 为前一个参数,在此代码中相当于n个参数
va_arg原型为:
type va_arg(va_list arg_ptr,type );//va_arg ,这个宏接受两个参数: va_list 变量和参数列表中下一个参数的类型,va_arg````返回这个参数的值,并使用 va_arg```指向下一个可变参数.
va_end原型为:
void va_end(va_list arg_ptr );

下面再来详细讲解一下这几个宏所代表的什么:
进行宏替换后的代码块:

#include <stdio.h>
#include <stdarg.h>
int average(int n, ...)//n个参数
{
    int sum = 0;  //和
    int i = 0;
    //va_list arg;  //创建arg变量
    char *  arg;
    //va_start(arg,n);
     ( arg = (char *)(&n) + 4 );

    for (i=0;i<n;i++)
    {
        //sum += va_arg(arg,int);//调用时分别取6,9,9
        sum += ( *(int *)((arg += 4) - 4) );
    }
    return sum/n;
    //va_end(arg);
     (arg = (char *)0 );
}
int main()
{
    int ret = average(3, 6,9,9);
    printf("ret = %d\n",ret);

    ret = average(5, 1,2,3,4,5);
    printf("ret = %d\n",ret);


    return 0;
}

vs 中源码 鼠标点击相应的宏转到定义为:

这里写图片描述

这里写图片描述

这里写图片描述

这里说明一下
_INTSIZEOF(int) //向上取整,当n传的字节大小为5,6,7,8,时宏返回的是8.即1-4时取4,5-8取8
_ADDRESSOF(n)  //顾名思义即 &n

各位读者如果有兴趣的话可以自行实现哦!
欢迎大家指教!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值