在Object-C中,咱们会遇到不少像NSLog这样的函数,其中参数的个数不肯定,由程序员自由控制,在初始化数组,字典等方面应用普遍,那么,这类的函数是如何实现的呢?咱们怎么编写咱们本身的省略参数的函数呢?固然,这不是惟一的多参函数的处理方法,你也能够经过一个字典或者数组传递参数。但C为咱们提供的这样的一种机制,无疑是最方便的。程序员
va_list数组
C语言中定义的一个指针,用于指向当前的参数。函数
va_start(ap,param)spa
这个宏是初始化参数列表,其中第一个参数是va_list对象,第二个参数是参数列表的第一个参数。3d
va_arg(ap, type)指针
一个用于取出参数的宏,这个宏的第一个参数是va_list对象,第二个参数是要取出的参数类型。code
va_end(ap)对象
这个宏用于关闭取参列表内存
在编写咱们本身的多参函数以前,明白函数的取参原理是十分重要的,首先,函数的参数是被放入咱们内存的栈段的,并且放入的顺序是从后往前放入,好比若是一个函数参数以下:it
void func(int a,int b,int c,int d)
那么传递参数的时候参数d先入栈,接着是c、b、a。如此这样,在取参的时候,根据堆栈的取值原则,则取值顺序为a、b、c、d。因此在原理上,只要咱们知道第一个参数的地址和每一个参数的类型,咱们就能够将参数都取出来。而上面介绍的几个宏,就是帮助咱们作这些的。
"..."这个符号就是咱们用来实现省略参数函数的符号。例如咱们模拟实现一个log函数以下:
-(void)myLog:(NSString *)str,...{//省略参数的写法 va_list list;//建立一个列表指针对象 va_start(list, str);//进行列表的初始化,str为省略前的第一个参数,及...以前的那个参数 NSString * temStr = str; while (temStr!=nil) {//若是不是nil,则继续取值 NSLog(@"%@",temStr); temStr = va_arg(list, NSString*);//返回取到的值,而且让指针指向下一个参数的地址 } va_end(list);//关闭列表指针 }
注意,调用时,咱们必须在参数的最后加上nil这个判断结束的条件:
[self myLog:@"312",@"321", nil];//必须有nil
细心的你可能发现了,这里的nil是咱们在调用函数时手动加上的,但是系统的许多函数在咱们调用时,系统直接帮咱们加上了参数结尾的那个nil,例如
NSArray * array = [NSArray arrayWithObjects:(id), nil]
这是如何作到的呢?咱们只须要在函数的声明里加上一个宏,就能够实现这个功能,修改以下:
-(void)myLog:(NSString *)str,...NS_REQUIRES_NIL_TERMINATION{//这里加上一个宏 va_list list; va_start(list, str); NSString * temStr = str; while (temStr!=nil) { NSLog(@"%@",temStr); temStr = va_arg(list, NSString*); } va_end(list); }
顾名思义,这个宏的做用就是在结束位置加上咱们须要的nil。
专一技术,热爱生活,交流技术,也作朋友。
——珲少 QQ群:203317592