va_end

C语言利用va_list、va_start、va_end、va_arg宏定义可变参数的函数

杀马特。学长 韩版系。学妹 提交于 2021-01-22 10:08:54
在定义可变参数的函数之前,先来理解一下函数参数的传递原理: 1、函数参数是以栈这种数据结构来存取的,在函数参数列表中,从右至左依次入栈。 2、参数的内存存放格式:参数的内存地址存放在内存的堆栈段中,在执行函数的时候,从最后一个(最右边)参数开始入栈。因此栈底高地址,栈顶低地址,举个例子说明一下: void test(int a, float b, char c); 那么,在调用test函数的时候,实参char c先进栈,然后是float b,最后才是int a,因此在内存中变量的存放次序是c->b->a,因为从理念上来说,我们只要探测到任意一个变量的地址,并且知道其它变量的类型,通过指针移位运算,就可以顺藤摸瓜找到其它的输入变量。 实现一个可变参数的函数,需要用到以下几个宏: typedef char* va_list; // 用于声明一个指向参数列表的字符型指针变量 void va_start(va_list ap, prev_param); // 第一个参数为指向可变参数字符指针变量,第二个参数是可变参数的第一个参数,通常用于指定可变参数列表中参数的个数 void va_arg(va_list ap, type); // 第一个参数为指向可变参数字符指针变量,第二个参数是可变参数的数据类型 void va_end(va_list ap);// 将存放可变参数字符串的变量清空

可变参数: va_start,va_arg,va_end

时间秒杀一切 提交于 2019-12-12 12:26:39
【推荐】2019 Java 开发者跳槽指南.pdf(吐血整理) >>> va_start,va_arg,va_end是在stdarg.h中被定义成宏的, 由于1)硬件平台的不同 2)编译器的不同,所以定义的宏也有所不同,下 面以VC++中stdarg.h里x86平台的宏定义摘录如下(’\’号表示折行): typedef char * va_list; #define _INTSIZEOF(n) \ ((sizeof(n)+sizeof(int)-1)&~(sizeof(int) - 1) ) #define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) ) #define va_arg(ap,t) \ ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) ) #define va_end(ap) ( ap = (va_list)0 ) 另外一种实现: typedef char * va_list;//va_list 等价于char*即字符指针。 #define va_start _crt_va_start//注意下面的替代。 #define va_arg _crt_va_arg #define va_end _crt_va_end #define _crt_va_start(ap,v)