美文网首页
iOS Tagged Pointer

iOS Tagged Pointer

作者: Q14 | 来源:发表于2019-08-27 21:03 被阅读0次

    Tagged Pointer 介绍

    苹果对于Tagged Pointer特点的介绍:

    Tagged Pointer主要解决内存浪费和访问效率的问题

    1. Tagged Pointer专门用来存储小的对象,NSStringNSNumberNSDateNSIndexPath
    2. Tagged Pointer指针的值不再是地址了,而是真正的值。所以,实际上它不再是一个对象了,它只是一个披着对象皮的普通变量而已。所以,它的内存并不存储在堆中,也不需要malloc和free。
    3. 在内存读取上有着3倍的效率,创建时比以前快106倍 销毁速度更快

    如果想深入了解

    如果你想要更进一步,去挖掘 Tagged Pointer 是如何实现的,可以参考 Friday Q&A 2012-07-27: Let’s Build Tagged Pointersobjc 源码

    为什么要引入Tagged Pointer

    iPhone5s 采用64位处理器。
    对于64位程序,我们的数据类型的长度是跟CPU的长度有关的。

    image.png

    这样就导致了 一些对象占用的内存会翻倍。

    同时 维护程序中的对象需要 分配内存,维护引用计数,管理生命周期,使用对象给程序的运行增加了负担。

    Tagged Pointer

    未引入 Tagged Pointer

    image.png

    引入 Tagged Pointer

    image.png

    判断是否是 Tagged Pointer


    image.png

    可以从 objc 源码中找出支持 Tagged Pointer 的对象类型,如下:

    typedef uint16_t objc_tag_index_t;
    enum
    {
        OBJC_TAG_NSString          = 2, 
        OBJC_TAG_NSNumber          = 3, 
        OBJC_TAG_NSIndexPath       = 4, 
        OBJC_TAG_NSDate            = 6, 
        ....
    };
    

    即针对 NSString、NSNumber、NSDate、NSIndexPath 这些类型,都支持 Tagged Pointer 技术。

    我们通过 NSNumber 以及 NSString 对象来观察 Tag+Data 的存储形式

        NSNumber *number1 = @1;                          //0xa061a1f0f2864c7a
        NSNumber *number2 = @2;                          //0xa061a1f0f2864c4a
        NSNumber *number3 = @(0xFFFFFFFFFFFFFFF);        //0x600003eac120
        NSNumber *number4 = @(1.2);                      //0x600003eac1c0
        int num4 = 5;
        NSNumber *number5 = @(num4);                     //0xa061a1f0f2864c3a 
        long num5 = 6;
        NSNumber *number6 = @(num5);                     //0xa061a1f0f2864c0b
        float num6 = 7;
        NSNumber *number7 = @(num6);                     //0xa061a1f0f2864c1c
        double num7 = 8;
        NSNumber *number8 = @(num7);                     //0xa061a1f0f2864ced
        
        //值: 0xa061a1f0f2864c7a 0xa061a1f0f2864c4a 0x600003eac120 0x600003eac1c0 0xa061a1f0f2864c3a 0xa061a1f0f2864c0b 0xa061a1f0f2864c1c 0xa061a1f0f2864ced
        NSLog(@"%p %p %p %p %p %p %p %p", number1, number2, number3, number4, number5, number6, number7, number8);
    

    由上表我们得出:

    • 很大的数字,超过 Tagged Pointer 表示上限的时候,将会转为对象存储,存放在堆上;
    • 如果是含有小数点的浮点数,将会直接以对象方式存储;
    • 其余类型的数字,包括不含小数部分的浮点型和整型都会以 Tagged Pointer 存储。
      并且,针对以上部分,我们整理出 Tagged Pointer 的存储格式如下,以 number1 为


      image.png

    NSString

    同上面 NSNumber 的处理逻辑,NSString 处理的类似。

    NSString *str1 = @"a";                                          //0x1049cc248
    NSString *str2 = [NSString stringWithFormat:@"a"];              //0xa000000000000611
    NSString *str3 = [NSString stringWithFormat:@"bccd"];           //0xa000000646363624
    NSString *str4 = [NSString stringWithFormat:@"c"];              //0xa000000000000631
    NSString *str5 = [NSString stringWithFormat:@"cdasjkfsdljfiwejdsjdlajfl"];//0x1c02418f0
    NSLog(@"%@ %@ %@ %@ %@",
          [str1 class],   //__NSCFConstantString
          [str2 class],   //NSTaggedPointerString
          [str3 class],   //NSTaggedPointerString
          [str4 class],   //NSTaggedPointerString
          [str5 class]);  // __NSCFString
    

    根据以上结果,我们将 NSString 分类三类:

    • 常量类型:__NSCFConstantString,定义的字符串常量。
    • Tagged Pointer 类型:NSTaggedPointerString,通过对象方法创建的短字符串。
    • NSString 对象类型:__NSCFString,包括 NSString、NSMutableString 等创建的字符串对象。

    以上,整理如下:

    image-20190103150923946

    image-20190103150923946

    NSString 以 Tagged Pointer 的存储格式如下:

    image-20190103151645748

    image-20190103151645748

    2.3 内存管理

    image-20190103151136639

    image-20190103151136639

    三、一个面试问题的研究

    该面试题如下:

    image-20190103151957866

    image-20190103151957866

    链接

    1. Friday Q&A 2012-07-27: Let’s Build Tagged Pointers
    2. Tagged Pointer wiki
    3. NSString retain count -1
    4. objc 源码
    5. 内存管理(二)Tagged Pointer

    相关文章

      网友评论

          本文标题:iOS Tagged Pointer

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