va_start的理解

va_start宏的功能是得到 第一个可变参数的地址

void test(char *format, ...) {

va_list ap; //char *ap; //ap用来保存第一个可变参数的地址

va_start(ap, format); //va_start展开之后就是利用固定参数(format)的地址得到紧随其后的第一个可变参数地址放入ap

//第一次展开: ap = format + (format占用的长度),通俗的讲就是由地址偏移长度得到下一个地址,这里format是个指针,所以format+format指针大小

//但是如果固定参数是个char或者short型,由于其长度为1或者2, 由于存在内存对齐(4的倍数),所以固定参数占用的长度需要再用一个宏

//第二次展开:

#define _INTSIZEOF(n)  ((sizeof(n)+sizeof(int)-1)&~(sizeof(int) - 1) )
参考:
~是位取反的意思。
_INTSIZEOF(n)整个做的事情就是将n的长度化为int长度的整数倍。
比如n为5,二进制就是101b,int长度为4,二进制为100b,那么n化为int长度的整数倍就应该为8。
~(sizeof(int) - 1) )就应该为~(4-1)=~(00000011b)=11111100b,这样任何数& ~(sizeof(int) - 1) )后最后两位肯定为0,就肯定是4的整数倍了。
(sizeof(n) + sizeof(int) - 1)就是将大于4m但小于等于4(m+1)的数提高到大于等于4(m+1)但小于4(m+2),这样再& ~(sizeof(int) - 1) )后就正好将原长度补齐到4的倍数了。

//结论就是:   利用 和可变参数相邻(之前)的固定参数的地址(栈中),偏移其占用的字节数(由于存在内存对齐,所以一个char或者short也会占用4个字节,所以需要要宏来计算其占用地址大小),就得到了可变参数的地址,MD,说来原理很简单的,但是看代码推原理就太费劲了。眼花缭乱的宏展开那真是要恶心死。还是搞懂原理比较好办
va_end(ap); //ap = NULL; //一般定义为 将ap指针设置为空,防止意外

}

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值