iOS开发中va_list的应用

va_list的定义:

VA_LIST 是在C语言中解决变参问题的一组宏,所在头文件:#include ,用于获取不确定个数的参数。
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的说明:

函数参数是存储在栈中的,函数参数从右往左依次入栈,参数的存储如下路

iOS开发中va_list的应用_第1张图片
va_list原理图.png

注意:

  • 可变参数的类型和个数完全由程序代码控制,它并不能智能地识别不同参数的个数和类型;
  • 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很傻,很不智能.......

你可能感兴趣的:(iOS开发中va_list的应用)