在函数中定义一个接受可变数量的参数,定义方式为:int func_name(int arg1,...);
在使用时代码抽象如下:
go
int func(int, ... ) {
.
.
.
}
int main() {
func(2, 2, 3);
func(3, 2, 3, 4);
}
- 需要使用 stdatg.h 头文件,该文件提供了实现可变参数功能的函数和宏
- 最后一个参数写成省略号,三个点。 int 表示要传递的可变参数总和。
- 使用步骤:
- 定义一个函数,最后参数为省略号,省略号可以设置自定义参数
- 在函数中创建一个 va_list 类型变量,该类型是在 stdarg.h 头文件中定义的
- 使用 int 参数和 va_start() 宏来初始化 va_list 变量为一个参数列表。 宏 va_start() 是在 stdarg.h 头文件中定义的
- 使用 va_arg() 宏 和 va_list 变量来访问参数列表中的每个项
- 使用宏 va_end() 来清理赋予 va_list 变量的内存
- 常见的宏使用
- va_start(ap , last_arg) 初始化可变参数列表。 ap 是 va_list 类型的变量,va_arg 是最后一个固定参数的名称,也可以是可变参数之前的参数。该宏将 ap 指向可变参数列表中的第一个参数
- va_arg(ap , type) 获取可变参数列表中下一个参数。ap 是va_list 类型的变量,type是下一个参数的变量。该宏返回类型为 type 的值,将 ap 指向下一个参数
- va_end(ap) 结束可变参数列表的访问。 ap 是va_list 类型的变量,该宏将 ap 设置为 NULL
写一个带有可变参数的函数,返回他们的平均值:
scss
#include<stdarg.h>
// 可变参数测试方法
double average(int num, ...)
{
va_list valist;
double sum = 0.0;
int i;
// 为 num 个参数初始化
va_start(valist, num);
//访问所有赋给 valist 的参数
for (i = 0; i < num; i++)
{
sum += va_arg(valist, int);
}
//清理为 valist 保留的内存
va_end(valist);
return sum / num;
}
printf("Average of 2,3,4,5 = %f \n", average(4, 2, 3, 4, 5));
printf("Average of 10,23,34,2,3,5 = %f\n", average(6,10,23,34,2,3,5));
其实看上面内容还很空洞,并没有理解。下面再参考一些网友的留言。
arduino
#include <stdio.h>
void debug_arg(unsigned int num, ...)
{
unsigned int i = 0;
unsigned int *addr = #
for (i = 0; i <= num; i++)
{
/* *(addr + i) 从左往右依次取出传递进来的参数,类似于出栈过程 */
printf("i=%d,value=%d\r\n", i, *(addr + i));
}
}
int main(void)
{
debug_arg(3, 66, 88, 666);
return 0;
}
下面是源代码, 进行拆分消化一下。
arduino
// 64 位机器用 8 字节对齐, 32 位 4 位对齐
#ifdef X64
#defin t long long
#else
#define t int
#endif
- 判断一下机器类型,64位机器用 8 字节对齐,32位 4 字节对齐。
arduino
//VA_LIST套宏中可以使用,用来改变INTSIZEOF中t的类型
//固定参数详见
void test(int a, double b, char* c)
{
char *p = (char*)&a;
//因为&a = void 类型 需要转换,void * =&a 不需要转换但是使用时要转换
printf("%p %p %p\n", &a, &b, &c);
//观察地址变化
printf("%p %s",(p+8),*(char**)(p+8+8));//64位机器时加8内存大小8字节对齐
return;
}
//可变参数实验
void test1(char* s,char *st,...)
{
char *ppt =(char*)&s;
//printf("%p %p %p %p,",ppt,&s,&st,(char*)ppt+8);
printf("%p %p %p %p\n", ppt, &s, &st, ppt + 8);
printf(" %d\n",*(int*)(ppt + 4+4));//当是X64就加8 X86就加4因为内存对齐规则
return;
}
int main()
{
char *p = "Hello world";
test1("111","eee",45234,23);
//test(2, 2.2, "Hello world");x
void *s = &p;
printf("%s", *(char**)s);
return 0;
}
其内涵主要在 va_list /start / arg/ end 的使用以及内存的理解。
此部分内容暂未完成,仍有很多需要理解的地方。