OC底层原理 07: 类的结构分析

主动已经是我对热爱东西表达的极限了

  • 类分析初探

通过LLDB查看类在内存中的分布情况

LLDB查看类的内存分布
  • 查看内存信息的三种方式:

    1.通过格式化输出当前类(x/4gx objc2),取类的isa首地址 & ISA_MASK得到类的指针地址po类的指针地址,得到当前类(objc2)x输出当前指针地址得到objc2类的内存信息

    1. 通过Class提供的API直接输出:x LGPerson.class得到类的内存信息

    2. 通过runtime提供的API直接得到类的内存信息,导入#import <objc/runtime.h>头文件,通过x object_getClass(objc2)得到类的内存信息

通过LLDB结果,发现0x00000001000020e80x00000001000020c0 都是objc2类,这个时候引入一个新的概念元类

  • 元类

    1.0x00000001000020c0isa获取类信息后,所指的类的isa的指针地址,称之为元类,我们说类的类为元类

    1. 类的方法归属都存在于元类
    2. 元类创建编译都是由编译器自动完成的

通过上述元类的理解,我们得知
0x00000001000020e8objc2isa 指针地址
0x00000001000020c0objc2元类

简单了解过元类,系统为什么要创建元类?元类与类有哪写区别?NSObject与元类又有哪些关联和区别?云类的存在又有哪些意义?

我们继续通过isa的走位一层一层来分析,使用LLDB查看类的内存信息元类的内存信息NSObject的内存信息如下图所示:

isa走向查看关联

  • isa走向描述:
    1. 通过obj2isa指针地址 & ISA_MASK得到元类的内存信息
    2. 通过元类isa指针地址 & ISA_MASK得到NSObject

根据isa走向可以得出如下结论:

  • isa 对象 -> 类(LGPerson) -> 元类(LGPerson) -> NSObject

在isa走向查看关联图中的最后,打印出了NSObject内存信息

(lldb) p/x NSObject.class
(Class) $18 = 0x0000000100333140 NSObject

为什么这里NSObject内存信息isa走向打印出来的NSObject内存信息不一致了?

isa走向NSObject内存信息:0x00000001003330f0
NSObject的内存信息:0x0000000100333140

  • 我们知道类的信息在内存中只存在一份,接下来我们开始验证类信息是否只存在一份

  • 方式一:通过不同形式定义,直接打印结果

//MARK: - 分析类对象在内存中存在个数
void TTWhetherOrNotTheOnly() {
    
    Class cls1 = [LGPerson class];
    Class cls2 = [LGPerson alloc].class;
    Class cls3 = object_getClass([LGPerson alloc]);
    NSLog(@"\n%p-\n%p-\n%p-",cls1,cls2,cls3);
}

//打印结果如下:
0x1000020f8-
0x1000020f8-
0x1000020f8-
  • 方式二:LLDB通过isa走向查看截图如下
isa走向
  • 方式三:runtime验证
//MARK: - 分析类对象在内存中存在个数3
void TTWhetherOrNotTheOnlyThree(){
    //NSObjec实例对象
    NSObject *obje3 = [NSObject alloc];
    //NSObject类
    Class clsss = object_getClass(obje3);
    //NSObject元类
    Class metaClass = object_getClass(clsss);
    //NSObject根元类(即NSObject)
    Class rootClass = object_getClass(metaClass);
    //NSObject根元类(即NSObject)
    Class rootClass1 = object_getClass(rootClass);
    NSLog(@"\n实例对象:-> %p\n NSObject类: -> %p\n NSObject元类: -> %p\n NSObject根元类(即NSObject): -> %p\n NSObject根元类(即NSObject):-> %p\n",obje3,clsss,metaClass,rootClass,rootClass1);
}

打印结果如下:

runtime验证

通过isa走向,发现最后NSObjectisa指针地址还是指向了NSObject

经典图来了
isa流程图

总结:

  • isa走向:
    实例对象(Instance of Subclass) -> 类(Class) -> 元类(meta Class) -> 根元类 NSObject (Root meta Class) -> 自己

  • superclass走向:
    类的继承关系:Class -> SuperClass -> RootClass -> nil
    元类的继承关系:meta Class -> SuperClass -> RootClass -> NSObject -> nil

  • 【注意】实例对象之间没有继承关系之间有继承关系

类的结构分析 objc_class & objc_object

通过对objc_classobjc_object对类进行深入分析,查看结构体在源码中的定义,代码如下:


struct objc_class : objc_object {
    // Class ISA; //8字节
    Class superclass;  //8字节
    cache_t cache;             // formerly cache pointer and vtable。(16字节)
    class_data_bits_t bits;    // class_rw_t * plus custom rr/alloc flags

      //bits 的 getter函数
    class_rw_t *data() const {
        return bits.data();
    }
      //bits 的setter函数
    void setData(class_rw_t *newData) {
        bits.setData(newData);
    }
    .
    .
    .
//  只复制的了部分,我们需要的,其他的可以自己去查看这里不错全部展示
}

查看objc2的内存信息

查看类信息
  • 说明:
    0x00000001000020d8 对应 objc_class 中的isa
    0x0000000100333140 对应 objc_class 中的 superclass
    0x000000010032d410 对应 objc_class 中的 cache
    0x0000801000000000 对应 objc_class 中的 bits

其中:
ISA表示继承于 objc_objectisa
superclass表示父类
cache表示缓存相关信息
bits里面存放类的相关数据(成员、属性、方法、协议等相关数据信息)

objc_classobjc_object在系统中的结构体定义如下:

typedef struct objc_class *Class;
typedef struct objc_object *id;
  • objc_class 与 objc_object 关系说明
    1. 结构体类型objc_class 继承自objc_object类型,其中objc_object也是一个结构体,且有一个isa属性,所以objc_class也拥有了isa属性

    2. NSObject 是一个类,用它初始化一个实例对象objc2objc2 满足objc_object 的特性(即有isa属性),主要是因为isa 是由 NSObjectobjc_class继承过来的,而objc_class继承自objc_objectobjc_objectisa属性。所以对象都有一个 isaisa表示指向,来自于当前的objc_object

    3. objc_object(结构体) 是 当前的 根对象所有的对象都有这样一个特性 objc_object,即拥有isa属性

【面试题】objc_object 与 对象的关系?

所有的对象 都是以objc_object为模板继承过来的

所有的对象 是 来自 NSObject(OC) ,但是真正到底层的 是一个objc_object(C/C++)的结构体类型

【总结】 objc_object对象关系继承关系

类的内存分布情况及验证

在探究类的内存分布之前,我们先了解一下什么是地址偏移

  • 地址偏移

定义如下代码:

int c[4] = {1,2,3,4};
int *d = c; //赋值c的地址给d

//打印出当前值所在的地址

ADLog(@"通过地址取:%p - %p - %p - %p - %p",&c,&c[0],&c[1],&c[2],&c[3]);

ADLog(@"通过便移取:%p - %p - %p - %p - %p",d,d+1,d+2,d+3,d+4);

       
for (int i=0; i<4; i++){
   int value = c[I];
   ADLog(@"%d",value);
}

打印结果如下:

打印截图

发现 &c&c[0]的地址是 相同的,都为 0x7ffeefbff500,因为c数组首地址指针代表当前c数组的地址指针所以与 &c[0]是一致的。c数组定义的为int类型的数据类型,所以相差的为4

下面我们通过地址偏移来取出数组c的值,通过LLDB打印C的值,截图如下:

LLDB数据偏移取出值
  • p *(地址) :打印地址获取值

我们通过数组c地址偏移 ,取出当前数组的
那么我们是不是也能通过地址偏移 来取出类的数据和所有的值
LGPerson为例:

通过地址偏移 取数组值的时候,我们知道需要偏移1 位就能够取出值,但是在类中如何偏移?
已知ISA的字节为8superclass的字节为8,但是cachebits的字节我们是未知的,要偏移多少位才能取出cachebits的值?

开始查看的cache所占字节数,进入cache的定义查看源码定义如下(这里只查看定义的字节大小的源码,源码太多不做展示):

struct cache_t {
#if CACHE_MASK_STORAGE == CACHE_MASK_STORAGE_OUTLINED
    explicit_atomic<struct bucket_t *> _buckets;  //8(结构体)
    explicit_atomic<mask_t> _mask;  //4 (内部定义的泛型所以为4)
#elif CACHE_MASK_STORAGE == CACHE_MASK_STORAGE_HIGH_16
    explicit_atomic<uintptr_t> _maskAndBuckets;
    mask_t _mask_unused;
      .
      .
      //省略代码,有兴趣可以自己去查看源码
      .
      .
#if __LP64__
    uint16_t _flags; //  2(unsigned类型)
#endif
    uint16_t _occupied; //2(unsigned类型)

      .
      .
      //省略代码,有兴趣可以自己去查看源码
      .
      .
}

通过cache的源码定义可以得知:cache的字节为16,这个时候我们就可以通过地址偏移得到bits里面存放类的相关数据(成员、属性、方法、协议等相关数据信息)

  • 查看bits信息

定义如下四个类:
LGDoctorLGPerson继承TTTeacher
TTPerson继承LGPerson

//--------------------------  TTTeacher(begin)   --------------------------
@interface TTTeacher : NSObject{
    
    NSString *TeacherOne;
}

@property (nonatomic, copy) NSString *teacherName;

@end
@implementation TTTeacher

@end
//--------------------------  TTTeacher (end)  --------------------------

//--------------------------  LGPerson(begin)   --------------------------
@interface LGPerson : TTTeacher{
    NSString *justOne;
}

@property (nonatomic, copy) NSString *name;

@end
@implementation LGPerson

@end
//--------------------------  LGPerson(end)   --------------------------

1.通过LLDB查看bits信息,截图如下:

bits数据信息

其中$1->data()是源码中调用了bits.data()方法;$2直接打印出bits信息

  • 【通过bits查看属性(properties)列表】
    查看class_rw_t在源码中的部分定义如下:
struct class_rw_t {  
    //  只是部分定义

  //成员变量
  const class_ro_t *ro() const {
        auto v = get_ro_or_rwe();
        if (slowpath(v.is<class_rw_ext_t *>())) {
            return v.get<class_rw_ext_t *>()->ro;
        }
        return v.get<const class_ro_t *>();
    }
    //获取方法
    const method_array_t methods() const {
        auto v = get_ro_or_rwe();
        if (v.is<class_rw_ext_t *>()) {
            return v.get<class_rw_ext_t *>()->methods;
        } else {
            return method_array_t{v.get<const class_ro_t *>()->baseMethods()};
        }
    }

    //  获取属性数组
    const property_array_t properties() const {
        auto v = get_ro_or_rwe();
        if (v.is<class_rw_ext_t *>()) {
            return v.get<class_rw_ext_t *>()->properties;
        } else {
            return property_array_t{v.get<const class_ro_t *>()->baseProperties};
        }
    }
    
    //协议
    const protocol_array_t protocols() const {
        auto v = get_ro_or_rwe();
        if (v.is<class_rw_ext_t *>()) {
            return v.get<class_rw_ext_t *>()->protocols;
        } else {
            return protocol_array_t{v.get<const class_ro_t *>()->baseProtocols};
        }
    }
};

LLDB命令bits获取属性properties操作查看如下:

bits属性查看.png
  • 【通过bits查看方法(methods)列表】

LLDB命令bits方法操作查看如下:

bits查看方法methods截图

  • 【通过bits查看方法(ro)列表】
bits查看成员变量

不详细阐述问题描述了,直接总结吧

  • 总结

通过bits存储信息,可以看出类的结构,类的数据存储
成员变量:bits->data() .ro() . ivars
方法:bits->data() . methods
属性:bits->data() .properties
可以查看出类的结构和数据存储位置,通过偏移查看信息
其中结构体使用.调用,对象方法使用->调用。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,126评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,254评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,445评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,185评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,178评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,970评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,276评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,927评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,400评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,883评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,997评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,646评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,213评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,204评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,423评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,423评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,722评论 2 345