va_list的定义:
VA_LIST 是在C语言中解决变参问题的一组宏,所在头文件:#include <stdarg.h>
,用于获取不确定个数的参数。
oc中的定义如下:
#ifndef _VA_LIST
typedef __builtin_va_list va_list;
#define _VA_LIST
#endif
#define va_start(ap, param) __builtin_va_start(ap, param)
#define va_end(ap) __builtin_va_end(ap)
#define va_arg(ap, type) __builtin_va_arg(ap, type)
宏:
INTSIZEOF:获取类型占用的空间长度,最小占用长度为int的整数倍:
#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
VA_START:获取可变参数列表的第一个参数的地址(ap是类型为va_list的指针,v是可变参数最左边的参数):
#define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) )
VA_ARG:获取可变参数的当前参数,返回指定类型并将指针指向下一参数(t参数描述了当前参数的类型):
#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
VA_END:清空va_list可变参数列表:
#define va_end(ap) ( ap = (va_list)0 )
va_list的说明:
函数参数是存储在栈中的,函数参数从右往左依次入栈,参数的存储如下路
注意:
- 可变参数的类型和个数完全由程序代码控制,它并不能智能地识别不同参数的个数和类型;
- va_start和va_end成对出现
va_list的应用:
当我们有一个需求:我们需要一个方法,但是入参个数不确定,这个时候就可以使用va_list
- (void)testMethod:(NSString *)string, ...NS_REQUIRES_NIL_TERMINATION{
va_list args;
if (string){
va_start(args, string);
NSString *otherstring = nil;
while ((otherstring = va_arg(args, NSString*))) {
NSLog(@"string = %@ point=%p",otherstring,otherstring);
}
va_end(args);
}
}
调用: [self testMethod:@"1",@"2",@"3",@(4),@"5",@"6",nil];
方法说明:
-
NS_REQUIRES_NIL_TERMINATION
告知编译器 需要一个结尾的参数,告知编译器参数的列表已经到最后一个不要再继续执行下去了。如果声明了在调用方法时如果没有更多的参数一定加上nil,否则会一直循环取出参数造成崩溃. -
va_start(args, string)
获取可变参数列表的第一个参数的地址,在这里是获取string的内存地址,这时args的指针 指向string -
va_arg(args, NSString*)
获取可变参数的当前参数,返回指定类型并将指针指向下一参数,第一次循环就会把指针指向上图的p2,下一次就会指向p3.直到结束,什么时候结束呢?如果我们不判断筛选条件,可以一直取出数据,我们通常直到内存空间取到数据为nil时或者0时结束。 -
va_arg(args, NSString*)
中会将指针指向下一参数,那应该偏移多少是到下一个参数地址呢?会等于与va_arg宏所返回的数值具有相同类型的对象的长度,比如:都是NSString类型,这时候偏移8字节,如果参数中混有其他类型的数据,如int,这时候偏移8字节能取到正确的数据吗,肯定是有问题的,所以当我们在传参数时,多个参数的类型要尽量一样
系统的NSLog也是用到va_list,va_list不会知道我们参数的数据类型和个数,这就是为什么我们需要占位符的原因
FOUNDATION_EXPORT void NSLog(NSString *format, ...) NS_FORMAT_FUNCTION(1,2) NS_NO_TAIL_CALL;
所以va_list很傻,很不智能.......