再论不定参数的问题

今天自己看了一下不定参数的文章,又深入理解了一下,总结如下:

 

#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!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值