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