函数样例:
#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