C语言不定长参数的实现 va_list

来源:互联网 发布:c语言中如何调用函数 编辑:程序博客网 时间:2024/05/22 06:37

1. C语言函数的调用方式 _cdecl 调用

_cdecl 是C Declaration的缩写(declaration,声明),表示C语言默认的函数调用方法:所有参数从右到左依次入栈,这些参数由调用者清除,称为手动清栈所以在函数调用栈中, 越右边的参数在栈的越低端,既内存地址越大。


2. 实现

va_list 是在C语言中解决变参问题的一组宏,所在头文件:#include typedef char* va_list

INTSIZEOF 宏,获取类型占用的空间长度,最小占用长度为int的整数倍:

#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )


VA_START宏,获取可变参数列表的第一个参数的地址(ap是类型为va_list的指针,v是可变参数最左边的参数):

#define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) )


VA_ARG宏,获取可变参数的当前参数,返回指定类型并将指针指向下一参数(t参数描述了当前参数的类型):

#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )


VA_END宏,清空va_list可变参数列表:

#define va_end(ap) ( ap = (va_list)0 )


#define _crt_va_start(ap,v) ( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) )

#define _crt_va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )

#define _crt_va_end(ap) ( ap = (va_list)0 )


3.用法  

void simple_va_fun(int i, ...)   

{    va_list arg_ptr;   

            int j=0;    

           va_start(arg_ptr, i);    

           j=va_arg(arg_ptr, int);    

          va_end(arg_ptr);    

          printf("%d %d\n", i, j);    

         return;   

}

0 0
原创粉丝点击