今天自己看了一下不定参数的文章,又深入理解了一下,总结如下:
#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
typedef char * va_list;
#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 )
第一眼看到_INTSIZEOF这个很不理解,不知道什么意思。
内事不懂问百度,于是搜了一下:
http://topic.csdn.net/u/20080719/10/824a4013-7a42-443c-abb6-dea0e0aa638e.html
现摘录关键的:
1:我们知道对于IX86,sizeof(int)一定是4的整数倍,所以~(sizeof(int) - 1) )的值一定是
右面[sizeof(n)-1]/2位为0,整个这个宏也就是保证了右面[sizeof(n)-1]/2位为0,其余位置为1
2:目的在于把sizeof(n)的结果变成至少是sizeof(int)的整倍数,这个一般用来在结构中实现按int的倍数对齐。
如果sizeof(int)是4,那么,当sizeof(n)的结果在1~4之间是,_INTSIZEOF(n)的结果会是4;当sizeof(n)的结果在5~8时,
_INTSIZEOF(n)的结果会是8;当sizeof(n)的结果在9~12时,_INTSIZEOF(n)的结果会是12;……总之,会是sizeof(int)的倍数。
剩下的一个自定义类型和三个宏也解释一下:
typedef char * va_list;
//va_list的类型就是char *,内存操作的基本单位是一个字节,下面将用 va_list来进行字节的操作
#define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) )
//将ap指针移动到参数v后面一个参数的开始位置
#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
//将ap指针移动到t类型参数后面一个参数的开始位置,返回t类型参数的值
#define va_end(ap) ( ap = (va_list)0 )
//将ap指针清空
以char *make_message(const char *fmt, ...);
make_message("Hello world %d %s", 5, endmsg)为例说明如下:
1:ap指针的操作是在函数的参数栈上操作的
2:可以通过&fmt获得函数参数栈第一个参数的地址
3:参数栈上保存参数时,如果sizeof(type)小于四,就是是四个字节的,大于四的是四的倍数,例如long long 是8
char类型,short类型,long类型,还有指针类型,将这些类型代到_INTSIZEOF宏中结果为4,long long类型结果为8
4:对于字符串类型,函数参数栈上保存的是指向字符串地址的指针,不是具体的字符串
将make_message函数参数栈打印出来(12个字节),和fmt的指针内容以及endmsg指针的内容比较可以看出来
基本就是这些,下面是测试程序,欢迎大家讨论,指正。
ps:上一篇中(不定参数),举的例子都是约定好了不定参数的类型的,在获取参数的时候,直接指定这个类型。
但是如果不知道不定参数的类型如何处理呢?就像printf一样。
下面是输出:
char 4
char * 4
int 4
short 4
name 4
long 4
long long 8
pointer of endmsg content is 0x4030aa
_INTSIZEOF fmt = 4
fmt pointer content is 0x403127
0x00000000 : 0x27 0x31 0x40 0x00 0x05 0x00 0x00 0x00 0xaa 0x30 0x40 0x00
Hello world 5 change yourself!