在看疯狂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
我们取n是6,int就占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
——我是太阳骑士索拉尔,愿阳光永照心中