1.va_list ap
用来保存宏va_start、va_arg和va_end所需要信息的一种类型。为了访问变长参数列表中的参数,必须声明va_list类型的一个对象。
va_list是一个链表指针,在iOS中被重定义。
2.va_start(ap, param)
这里的函数作用是将参数列表的第一个参数的地址给我们之前定义的参数链表指针给赋值,用于系统进行遍历取值。ap--我们之前定义的偏移指针 param--参数列表的第一个参数
3.va_arg(ap, type)
va_arg函数的作用是根据指针进行取值,取出值以后返回,并且指针偏移一位,ap---作用同上 type--参数的类型
4.va_end(ap)
参数列表遍历完毕后我们需要将之前定义的指针偏移量给销毁,以防出现意外。这里属于安全操作
你一定会时常见到有些方法的最后会有一个无关痛痒的 nil 参数,可曾想过为什么要带这个参数呢。
解答:因为这里参数的传递是可变参数的传递,
拿[NSArray arrayWithObjects:]为例子,我们进入他的接口文件去看会看到接口的实现是这样的
+ (instancetype)arrayWithObjects:(ObjectType)firstObj, ... NS_REQUIRES_NIL_TERMINATION;
最后的参数是一个宏定义,字面的意思是NS系列的‘需要以nil作为终止符‘
这里由于是可变参数,所以参数的个数并不确定,所有在系统进行遍历该你传进去的参数时会进行判断是否读取到nil的终止符。当读到终止符nil时确定参数的个数停止进行遍历
(这也就是为什么当数组中对象是nil值的时候,后面的值就加不进去的原因。所以数组的初始化尽量使用字面量形式,字面量可以导致错误展现出来。不至于不知道哪里错了)
由此,我们可以自己进行方法的设定时,也能实现可变参数的实现(Java的可变参数比objective-c要容易的多),实现如下
1 声明 2 3 - (void)test:(NSString *)first,...NS_REQUIRES_NIL_TERMINATION;
1 实现 2 3 //可变参数 4 5 - (void)test:(NSString *)first,...{ 6 7 //参数链表指针 8 9 va_list list; 10 11 //遍历开始 12 13 va_start(list, first); 14 15 //知道读取到下一个时nil时结束递增 16 17 for (NSString *str = first; str != nil; str = va_arg(list, NSString*)) { 18 19 NSLog(@"%@",str); 20 21 } 22 23 //结束遍历 24 25 va_end(list); 26 27 }
在讲他的实现原理的时候我们不得不牵扯C语言中可变参数的实现原理:C语言中的参数,编译器会将多个参数从右向左挨个入栈,然后挨个出栈,所以只要我们知道参数列表中的第一个或者最后一个,就能够挨个把他们取出来。
原文链接:http://www.jianshu.com/p/4e43d5478180
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。