不定参数的分析

先看一个简单的使用例子

求任意个自然数的平方和:

首先解释下函数参数入栈情况

在VC等绝大多数C编译器中,默认情况下,参数进栈的顺序是由右向左的,因此,参数进栈以后的内存模型如下图所示:最后一个固定参数的地址位于第一个可变参数之下,并且是连续存储的。

| 最后一个可变参数(高内存地址处) | 第N个可变参数  | 第一个可变参数 | 最后一个固定参数 | 第一个固定参数(低内存地址处) 
明白上面那个顺序,就知道其实可变参数就是玩弄参数的地址,已达到“不定”的目的

下面我摘自VC中的源码来解释va_list,va_start,va_arg,va_end宏

 

1.其实va_list就是我们平时经常用的char*

typedef char *  va_list;

2.va_start该宏的目的就是将指针指向最后一个固定参数的后面,即第一个不定参数的起始地址

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

v即表示最后一个固定参数,&v表示v的地址,

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

该宏其实是一个内存对齐的操作。即表示大于sizeof(n)且为sizeof(int)倍数的最小整数。这句话有点绕,其实举几个例子就简单了。比如1--4,则返回4,5--8则返回8
3.va_arg 该宏的目的是将ap指针继续后移,读取后面的参数,t表示参数类型。该宏首先将ap指针移动到下一个参数的起始地址ap += _INTSIZEOF(t),然后将本参数的值返回

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

4.va_end将指针赋空
#define va_end(ap) ap = (va_list)0

 

有了这个分析我们可以把上例中的代码重新翻译下

同样我们也可以写出我们自己的printf了

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值