OC 类对象Class探索

类对象本质为objc_class结构体。类对象⾥⾯存储了类的⽗类、属性、实例⽅法、协议、成员变量、⽅法缓存等等

struct objc_class : objc_object {

    // Class ISA; // 8 objc_object 结构体属性

    Class superclass;// 8

    cache_t cache;  // 16 

    class_data_bits_t bits;

...

}

获取实例对象的类对象 runtime API

Class class1 = [Person class];

    Class class2 = [Person alloc].class;

    Class class3 = object_getClass([Person alloc]);

isa 探索

void TestNSObject(void){

    // NSObject实例对象

    NSObject *object1 = [NSObject alloc];

    // NSObject类对象

    Class class =object_getClass(object1);

    // NSObject元类(根元类)

    Class metaClass = object_getClass(class);


    NSLog(@"NSObject实例对象:%p",object1);

    NSLog(@"NSObject类对象:%p",class);

    NSLog(@"NSObject元类(根元类):%p",metaClass);


    // Person  -- 元类的父类就是父类的元类

    Class pMetaClass = objc_getMetaClass("Person");

    Class psuperClass =class_getSuperclass(pMetaClass);

    NSLog(@"%@ - %p",pMetaClass,pMetaClass);

    NSLog(@"%@ - %p",psuperClass,psuperClass);


    // Teacher继承自Person

    // Teacher元类的父类 就是 Person(Person的元类)

    Class tMetaClass =objc_getMetaClass("Teacher");

    Class tsuperClass =class_getSuperclass(tMetaClass);

    NSLog(@"%@ - %p",tsuperClass,tsuperClass);


    // NSObject的父类

    Class nsuperClass = class_getSuperclass(NSObject.class);

    NSLog(@"%@ - %p",nsuperClass,nsuperClass);


    // 根元类的父类 -- NSObject

    Class rnsuperClass =class_getSuperclass(metaClass);

    NSLog(@"%@ - %p",rnsuperClass,rnsuperClass);

}

实例对象: isa + 成员变量的值

isa 是一个nonpointerIsa,其中几位是类对象的内存地址,所以

isa 走向 实例对象 isa -->类对象 isa -->元类对象 isa -->根元类 isa -->根元类自己 --

元类的继承关系

⽗类的元类就是元类的⽗类。根元类的⽗类就是NSObject。NSObject是万类之祖。

class_data_bits_t 存储方法,属性,成员变量

成员变量存放在类对象的class_ro_t结构体当中

lldb调试

(lldb) x/6gx p.class

0x100008250: 0x0000000100008228 0x000000010080e140

0x100008260: 0x0000000100d6a430 0x0002802c00000003

0x100008270: 0x0000000100b48cc4 0x0000000000000000

(lldb) p/x (class_data_bits_t *)0x100008270

(class_data_bits_t *) $1 = 0x0000000100008270

(lldb) p $1->data()

(class_rw_t *) $2 = 0x0000000100b48cc0

(lldb) p *$2

(class_rw_t) $3 = {

  flags = 2148007936

  witness = 1

  ro_or_rw_ext = {

    std::__1::atomic = {

      Value = 4295000512

    }

  }

  firstSubclass = nil

  nextSiblingClass = 0x00007ff8592549b8

}

(lldb) p $3.methods()

(const method_array_t) $4 = {

  list_array_tt = {

     = {

      list = {

        ptr = 0x0000000100008098

      }

      arrayAndFlag = 4295000216

    }

  }

}

(lldb) p $4.list

(const method_list_t_authed_ptr<method_list_t>) $5 = {

  ptr = 0x0000000100008098

}

(lldb) p $5.ptr

(method_list_t *const) $6 = 0x0000000100008098

(lldb) p *$6

(method_list_t) $7 = {

  entsize_list_tt = (entsizeAndFlags = 27, count = 6)

}

(lldb) p $7.get(0)

(method_t) $8 = {}

(lldb) p $8.getDescription()

(objc_method_description *) $9 = 0x00000001000080a0

(lldb) p *$9

(objc_method_description) $10 = (name = "instanceMethod", types = "v16@0:8")

(lldb) p $7.get(5).getDescription()

(objc_method_description *) $11 = 0x0000000100008118

(lldb) p *$11

(objc_method_description) $12 = (name = "setAge:", types = "v20@0:8i16")

(lldb) p $7.get(4).getDescription()

(objc_method_description *) $13 = 0x0000000100008100

(lldb) p *$13

(objc_method_description) $14 = (name = "age", types = "i16@0:8")

(lldb) p *($7.get(3).getDescription())

(objc_method_description) $15 = (name = "setName:", types = "v24@0:8@16")

(lldb) p *($7.get(2).getDescription())

(objc_method_description) $16 = (name = ".cxx_destruct", types = "v16@0:8")

.cxx_destruct⽅法是在ARC模式下⽤于释放成员变量的。只有当前类拥有实例变量时这个⽅法才会出现,property⽣成的实例变量也算,且⽗类的实例变量不会导致⼦类拥有这个⽅法。

苹果为什么设计元类?

能够复⽤消息传递这套机制。不管你是什么类型的⽅法,都是同⼀套流程

ro,rw,rwe区别

ro 编译时生成,存类的属性,实例⽅法,协议,不可修改

class_ro_t是在编译的时候⽣成的。当类在编译的时候,类的属性,实例⽅法,协议这些内容就存在class_ro_t这个结构体⾥⾯了,这是⼀块纯净的内存空间,不允许被修改

struct class_rw_t {

...

   explicit_atomic<uintptr_t> ro_or_rw_ext

...

private:

    using ro_or_rw_ext_t = objc::PointerUnion<const class_ro_t, class_rw_ext_t, PTRAUTH_STR("class_ro_t"), PTRAUTH_STR("class_rw_ext_t")>;

...

}

rw 结构体能访问到 ro和 rwe

class_rw_t是在运⾏的时候⽣成的,类⼀经使⽤就会变成class_rw_t,它会先将class_ro_t的内容"拿"过去,然后再将当前类的分类的这些属性、⽅法等拷⻉到class_rw_t⾥⾯。它是可读写的。

 const method_array_t methods() const {

        auto v =get_ro_or_rwe();

        if(v.is<class_rw_ext_t *>()) {

            return v.get(&ro_or_rw_ext)->methods;

        }else{

            return method_array_t{v.get<const class_ro_t*>(&ro_or_rw_ext)->baseMethods};

        }

    }

获取方法时,从class_rw_ext_t 取 或 从class_ro_t取

class_rw_ext_t可以减少内存的消耗。苹果在wwdc2020⾥⾯说过,只有⼤约10%左右的类需要动态修改。所以只有10%左右的类⾥⾯需要⽣成class_rw_ext_t这个结构体。这样的话,可以节约很⼤⼀部分内存。

class_rw_ext_t⽣成的条件:

⽤过runtime的Api进⾏动态修改的时候。或有分类的时候,且分类和本类都为⾮懒加载类(实现了+load⽅法)的时候。

rw,ro,rwe

cache_t 缓存

⽅法的缓存基于不同架构,缓存策略是不⼀样的。以下是⽅法缓存的核⼼代码的部分截图,这⾥就体现了在不同架构下的不同的缓存策略。

缓存核心代码

在arm64结构,也就是真机环境下,刚开始初始化的缓存⽅法的容器的⻓度2,当容器的⻓度⼩于8时,是满容量了才扩容。当容器的⻓度⼤于8时,是7/8扩容。也就是说当容器的⻓度为8时,容器可以存储8个⽅法。当容器的⻓度为16时,当第15个⽅法需要存储进来的时候,容器就要扩容了。

在x86_64架构下,刚开始初始化的容器的⻓度为4,是3/4扩容。这⾥的3/4扩容指的是:如果容器的⻓度为4,当第3个数据需要存储的时候,就要扩容了。如果容器的⻓度为8,当第6个数据需要存储的时候,就要扩容了。也就是说容器只能存储容器⻓度的3/4减1个⽅法。

还有⼀点就是:当容器扩容之后,前⾯存储的⽅法也会随之清空

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

推荐阅读更多精彩内容