iOS开发中va_list的应用

作者: Dely | 来源:发表于2017-09-05 14:46 被阅读154次

    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_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的应用

      本文链接:https://www.haomeiwen.com/subject/sfjujxtx.html