可变参数函数(二)

函数样例:

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

double add(int n,...)
{
    int i = 0;
    double sum = 0;
    va_list argptr;
    va_start(argptr,n);
    for(i = 0 ; i < n; i++)
    {
        double d =  va_arg(argptr,double);
        printf("%d argument is %.3f
",i,d);  //获取第i个double型的可变参数
        sum += va_arg(argptr,double);        
    }
    va_end(argptr);    
    return sum;
}

int main(int argc,char **argv)
{
    double sum = 0;
    float *p = malloc(argc * sizeof(float));
    int i;
    for(i = 1; i < argc; i++)        
    {        
        p[i] = atof(argv[i]);    
        //printf("p[%d] = %.3f
",i,p[i]);
    }

    sum = add(1,p[1]);    
    printf("sum = %.3f
",sum);
    sum = add(2,p[1],p[2]);    
    printf("sum = %.3f
",sum);
    sum = add(3,p[1],p[2],p[3]);    
    printf("sum = %.3f
",sum);

    free(p);
    return 0;
}

运行结果:

exbot@ubuntu:~/wangqinghe/DeBug/20190702$ gcc sum.c -o sum

exbot@ubuntu:~/wangqinghe/DeBug/20190702$ ./sum 10 20 30

0 argument is 10.000

sum = 0.000

0 argument is 10.000

1 argument is 0.000

sum = 20.000

0 argument is 10.000

1 argument is 30.000

2 argument is 0.000

sum = 20.000

测试:double d =  va_arg(argptr,double);是否只提取可变参数列表中double类型的数据

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

double add(int n,...)
{
    int i = 0;
    double sum = 0;
    va_list argptr;
    va_start(argptr,n);
    for(i = 0 ; i < n; i++)
    {
        double d =  va_arg(argptr,double);
        printf("%d argument is %.3f
",i,d);
        sum += va_arg(argptr,double);        
    }
    va_end(argptr);    
    return sum;
}

int main(int argc,char **argv)
{
    double sum = 0;
    float *p = malloc(argc * sizeof(float));
    int i;
    for(i = 1; i < argc; i++)        
    {    
        if(0 == i % 2)    
        {
            p[i] = atof(argv[i]);
            printf("float : p[%d] = %.3f
",i,p[i]); 
        }
    }

    sum = add(1,p[1]);    
    printf("sum = %.3f
",sum);
    sum = add(2,p[1],p[2]);    
    printf("sum = %.3f
",sum);
    sum = add(3,p[1],p[2],p[3]);    
    printf("sum = %.3f
",sum);

    free(p);
    return 0;
}

运行结果:

exbot@ubuntu:~/wangqinghe/DeBug/20190702$ ./sum 10 20 30 40 50 60 70

float : p[2] = 20.000

float : p[4] = 40.000

float : p[6] = 60.000

0 argument is 0.000

sum = 0.000

0 argument is 0.000

1 argument is 0.000

sum = 20.000

0 argument is 0.000

1 argument is 0.000

2 argument is 0.000

sum = 20.000

结论:可变参数的输入类型要和使用的一致。(暂定)。


函数介绍:

va_list

这是一个适用于 va_start()、va_arg() 和 va_end() 这三个宏存储信息的类型。

一般情况下va_list所定义变量为字符指针,即typedef char *va_list

该类型的变量用于存储参数的地址。因为得到参数的地址之后,再结合参数的类型,才能得到参数的值。

va_start

宏原型是:

void va_start(va_list ap, last_arg)

参数:

ap是一个 va_list 类型的对象,它用来存储通过 va_arg 获取额外参数时所必需的信息。

last_arg是最后一个传递给函数的已知的固定参数。 

该宏与 va_arg 和 va_end 宏是一起使用的,且必须在使用 va_arg 和 va_end 之前被调用。

va_arg

宏原型是:

type va_arg(va_list ap, type) 

其作用是从ap开始取一个type型的值返回,并且自动将ap指向下一个参数。所以如果参数类型写错了,例如将char*写成char了,本来要取4个字节,结果只取了一个字节,ap本来要向后面移动4个字节,结果只移动了一个字节,后面的数据就全错了。

参数:

ap是一个 va_list 类型的对象,存储了有关额外参数和检索状态的信息。该对象应在第一次调用 va_arg 之前通过调用 va_start 进行初始化。

type这是一个类型名称。该类型名称是作为扩展自该宏的表达式的类型来使用的。 

返回值:该宏返回下一个额外的参数,是一个类型为 type 的表达式。

注意:

va_arg宏的第2个参数不能被指定为char、short或者float类型。

因为char和short类型的参数会被转换为int类型,而float类型的参数会被转换为double类型。如果错误的指定了,将会在程序中引起麻烦。

例如,这样写肯定是不对的:

c = va_arg(ap,char);

因为我们无法传递一个char类型参数,如果传递了,它将会被自动转化为int类型。上面的式子应该写成:

c = va_arg(ap,int);

va_end

宏原型为:

 void va_end(va_list ap) 

参数:

ap是之前由同一函数中的 va_start 初始化的 va_list 对象。 

其作用是作用是将ap设置为NULL,如果在从函数返回之前没有调用 va_end,则结果为未定义。

总结

使用可变长参数的步骤:

1.声明va_list变量;

2.使用va_start指定可变长参数的位置;

3.使用va_arg来获取参数值;

4.可选,使用va_end将va_list清零。

参考原文:https://blog.csdn.net/hmxz2nn/article/details/80445770

原文地址:https://www.cnblogs.com/wanghao-boke/p/11121054.html