printf其实就是使用了可变参数的函数,printf的原型定义为
int printf(char *format,...);
我们在使用printf的时候可以这样
ptintf("hello");
ptintf("%s ,today is % year","hello",2011);
可变参数由调用函数动态的传递参数进被调函数(被调函数使用...声明自己支持可变参数),被调函数通过格式字符串来识别调用函数传递进来的参数个数和类型
可变参数解析
我们知道C程序函数调用的时候是遵循__stdcall,也就是调用者负责参数的压栈和清理,参数在栈上是按进栈顺序排列,__stdcall指定函数调用表达式最右边的参数最先进栈,而最左的的参数最迟进栈,因此如果栈是向下(高地址向低地址)增长的话,最迟进栈的参数在栈的最下方,之前进栈的参数通过位置偏移(+4)就可以找到。所以定义一个指针指向到最低或最高的参数,其他参数都可以依次找到。
#include <stdarg.h>
void g(int i, ...)
{
//char *arg_ptr;
va_list arg_ptr;
int j=0,k=0;
va_start(arg_ptr, i);
//arg_ptr = (char*)&i;
//arg_ptr =arg_ptr + sizeof(i);
j=va_arg(arg_ptr, int);
//j = *(int*)arg_ptr;
//arg_ptr += sizeof( *(int*)arg_ptr);
k=va_arg(arg_ptr, int);
//k = *(int*)arg_ptr;
//arg_ptr += sizeof( *(int*)arg_ptr);
va_end(arg_ptr);
//arg_ptr = 0;
printf("%d %d %d\n", i, j, k);
return;
}
g(100,200,300);
VS CRT 宏
#define _ADDRESSOF(v) ( &(v) ) //C语言版本
#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
#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 )
我自定义的宏应该可以替代原有宏
#define va_list my_va_list
#define my_va_list char*
#define va_start my_va_start
#define va_arg my_va_arg
#define va_end my_va_end
#define my_va_start(ap,v) ap = (va_list)&v;ap = ap + sizeof(v)
#define my_va_arg(ap,t) *(t*)ap;ap += sizeof( *(t*)ap)
#define my_va_end(ap) ap = 0