一、使用场景
有时候我们需要自定义一个方法来实现一些功能,(比如:暴打小卷),我们的方法里面需要指定谁来暴打小卷,打小卷的人数不确定,打小卷的是不是人也无法确定,所以我们需要用到可变参数。
二、简单介绍
简单来说,可变参数就是我们在自定义方法的时候,传递的参数类型,参数个数可以根据需要来改变。实现像下面这样去调用这个方法:第一个参数固定,其它参数类型不同,参数个数也不一样。当然,我们定义两个方法也可以实现,但是如果李成鹏每次打小卷带的小弟都不一样的话呢?
[self 暴打小卷 : 李成鹏,王冬,尉超,猫,狗,nil];
[self 暴打小卷 : 李成鹏,张洪林,王鼎,猪,nil];
三、可变参数的实现原理
1.在C中,当我们无法列出传递函数的所有实参的类型和数目时,可以用省略号指定参数表,OC中自然也可以这么玩:
- (void)暴打小卷:(人 *)李成鹏, ...;
2.函数的参数在内存中是以栈的形式存储的,存储顺序是按照参数从右往左的顺序依次入栈,根据栈的先进后出的特性,接下来取参数的时候就是从左向右的
顺序依次取值了。从理论上讲,只要取到了其中一个参数的地址,“顺藤摸瓜"总能找到其它的参数地址。所以,需要将第一个参数固定。根据第一个参数的地址,来取到其它的参数。
3.在写具体的实现方法之前,我们需要用到C库里面的几个”工具“来”暴打小卷“(动家伙,打扁他^_^)。这些工具有:刀、枪、棍、棒。。。(⊙o⊙)…其实是:
typedef char * va_list :这是一个字符类型的指针,指针指向当前的参数,需要通过这个指针取参数。
void va_start(va_list ap,param) :初始化字符指针的函数,将指针指向方法中的第一个可变参数,这个函数需要两个参数:va_list和自定义的方法中固定的那个参数(比如暴打小卷中的“李成鹏”)
type va_arg(va_list ap,type):这是一个取参数的函数,这个函数需要两个参数,第一个参数是字符指针va_list,第二个参数是我们需要取的可变参数的数据类型。这个函数需要做两件事:1:取到指定数据类型的参数。2:将指针ap指向下一个可变参数的地址。
void va_end(va_list ap):当我们取完所有的可变参数之后,需要将指针ap指向NULL。不然它就可能吓JB乱指,就成了一个野孩子:野指针。这个函数需要和va_start成对使用。
4.通过上面第三条说的,其实实现可变参数的步骤已经显而易见了:
1、定义一个va_list指针,用来指向打小卷的凶手们。警察开始立案调查%>_<%。
2、调用 va_start 初始化这个指针,首先指向打小卷的幕后主使(李成鹏)抓到他之后就能抓到其他凶手。
3、调用 va_arg 开始取参数,主使抓到了,小弟们自然跑不掉。
4、调用 va_end 将va_list指针置空。可以结案了O(∩_∩)O~~。
四、具体实现
不多说,直接上代码实例:
// 定义一个方法
-(void)暴打小卷:(人*)主使,...{
// 定义指针
va_list ap;
// 开始取值,指针先确定主使人
va_start(ap,主使);
NSLog(@"打小卷的主使是:%@",主使);
// 循环取值
while(YES){
// 因为不确定参数的类型,所以指定数据类型是id类型,并且用id类型的变量来接收。
id obj=va_arg(ap,id);
// 取完所有参数之后,跳出循环
if(obj==nil)break;
NSLog(@"打小卷的帮凶有:%@",obj);
}
// 取完之后毁掉va_list指针
va_end(ap);
}
调用:简单介绍里面已经写了调用的示例,需要注意的是,参数的结尾必须写上nil来让取参数的时候的循环停止。当然机智的你或许有其他的办法来控制循环次数,就不需要写最后的nil了。比如对sql语句的?进行赋值的时候,可以判断问号的个数来控制循环次数。