关于可变参数函数+ _INTSIZEOF宏实现详细

在看疯狂iOS讲义,讲到形参个数可变的问题(p 155)用到了下面三个函数,我直接把函数原型找出来了


void va_start( va_list arg_ptr, prev_param ); 

type va_arg( va_list arg_ptr, type ); 

void va_end( va_list arg_ptr );


其中 va_list arg_ptr是定义了一个list型的变量,该变量是指向参数的指针


这些宏定义在stdarg.h va_start的第二个参数是第一个可变参数的前一个参数,是一个固定参数,


va_arg返回可变参数,第二个参数是你要返回的参数的类型


va_end宏结束可变参数的获取,然后你就可以在参数中使用第二个参数。若果有多个可变参数的,一次调用va_arg获取各个参数

下面是函数的定义,可以看到用了一个特别的宏:

typedef char * va_list; 

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

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

sizeof(n) “向上取整”成 sizeof(int) 的整数倍,用来地址对齐。


~的意思是按位取反,&的意思按位取与,就是除了都是两个数都是1,其他都是得0


我们取n6int就占4个字节,前半段:6+4-1=9

对于任何数,我们加上一个三,如果他是4的倍数,不会被不到下一个4的倍数,如果不满4的话,加上三补齐凑够4的倍数

然后看后半段有4-1=3,取二进制为(为了方便,就画八位):
0000 0011

再取反:

1111 1100

而此时,我们前半段得出的9的二进制为:

0000 1001

这两个数一取与,可以看到最后一个1被抹掉了

0000 1000

得出8了(就是辣么神奇)


参考或深入文章:

https://blog.csdn.net/u010476094/article/details/39527697

https://blog.csdn.net/edonlii/article/details/8497704


——我是太阳骑士索拉尔,愿阳光永照心中

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值