ObjC runtime源码(一):Objective-C Class, Class cluster(类族)

ObjC的runtime只能在Mac OS下才能编译, 代码都是在x86_64架构下运行,iOS上是在arm64, armv7s, armv7架构下运行

遇到的问题

最近在做iOS上的闪退防御,最常见的就是防御NSMutableDictionarynil值闪退,使用的方法也很简单,就是swizzleNSMutableDictionary里的setObject:forKey:方法,但是试了很多次都没有swizzle到该方法,觉得很诡异,结果找了很多资料才发现,原来在[self class]上swizzlesetObject:forKey:这个方法是没有用的,因为在NSMutableDictionary的扩展中[self class]返回的是NSMutableDictionary, 在项目中创建的NSMutableDictionary其实都是NSMutableDictionary的子类__NSDictionaryM,因为NSMutableDictionary是一个类族。这篇文章就在runtime源码的背景下讲下Objective-C的类和类族

// 不生效的方法
@implementation NSMutableDictionary (YITCrashGuard)

+ (void)swizzleDictionary {
    [YITCrashGuardSwizzler swizzleSelector:@selector(setObject:forKey:) withSelector:@selector(swizzled_setObject:forKey:) onClass:[self class]];
}
// 生效的方法
@implementation NSMutableDictionary (YITCrashGuard)

+ (void)swizzleDictionary {
    [YITCrashGuardSwizzler swizzleSelector:@selector(setObject:forKey:) withSelector:@selector(swizzled_setObject:forKey:) onClass:objc_getClass("__NSDictionaryM")];
}

Objective-C的类,类的实例的实现方式

  1. x86_64, i386, arm64, armv7s, armv7的区别

    • x86_64:针对x86架构的64位处理器需要的架构
    • i386:针对intel通用32位处理器需要的架构
    • x86_64, i386是Mac处理器的指令集
    • arm64:64位iPhone真机上处理器需要的架构,iPhone 5s以上
    • armv7s:32位iPhone真机上处理器需要的架构,iPhone 5, iPhone 5c
    • armv7:32位iPhone真机上处理器需要的架构,iPhone 5以下
    • arm64, armv7s, armv7是ARM处理器的指令集
  2. 先看下Objective-C中类的实现,在objc.h文件中可以找到Objective-C中类跟类的实例的描述

    • Class(类):
      /// An opaque type that represents an Objective-C class.
      typedef struct objc_class *Class;
      
    • instance of a Class(对象):
      /// A pointer to an instance of a class.
      typedef struct objc_object *id;    // id就是我们平时遇到的id类型
      
  3. 由此可见,Objective-C对象都是C语言的结构体(struct),现在我们看下NSObject的实现,在NSObject.h文件可以找到:

    @interface NSObject <NSObject> {
        Class isa  OBJC_ISA_AVAILABILITY;
    }
    

    代码中的Class即为objc_class的结构体

  4. 那么objc_class, objc_object又是怎么定义的呢?

    • objc-runtime-new.h文件中可以找到objc_class的定义
      struct objc_class : objc_object {
          // Class ISA;
          isa_t isa;                 // inherited from objc_object
          Class superclass;
          cache_t cache;             // formerly cache pointer and vtable
          class_data_bits_t bits;    // class_rw_t * plus custom rr/alloc flags
          ...
      };
      
      • isa: 继承于objc_object
      • superclass: 指向自己父类的指针
      • cache: 方法缓存
      • bits: 类的实例方法链表
    • objc-private.h文件中可以找到objc_object的定义
      struct objc_object {
      private:
          isa_t isa;
      
      public:
      
          // ISA() assumes this is NOT a tagged pointer object
          Class ISA();
      
          // getIsa() allows this to be a tagged pointer object
          Class getIsa();
          ...
      };
      
  5. 由此可见,objc_class继承于objc_object, 所以也是包含一个isa结构体的,总结下来就是说,在Objective-C里,不只是对象的实例包含一个isa结构体,这个对象的类本身也有这么一个isa,所以说白了,其实Objective-C里的类也是一个对象


元类(meta-class)

  1. 如果每个对象都保存了自己能执行的方法,对内存的占用会有很大的影响,所以在Objective-C中,对象的方法是不会存在对象的结构体(objc_object)中的。当一个对象的实例方法被调用时,这个对象的实例需要通过自己持有的isa来查找它自己对应的类(objc_class),在objc_class中的class_data_bits_t结构体中查找对应方法的实现,找不到的话,objc_class会通过superclass查找继承的方法。

  2. 对象的实例方法是通过class_data_bits_t结构体来查找,那么类方法又是怎么查找并调用的呢?这个时候就是我们要说的元类了,元类可以保证不管是类还是对象都可以通过相同的机制查找方法的实现

    Runtime-2.png

    • 实例方法调用时,通过对象的isa在Class中获取方法的实现
    • 类方法调用时,通过类的isa在meta-class中获取方法的实现
      objc-isa-class-diagram.jpg
  3. [Class class], [obj class], object_getClass(Class)的区别, class方法是得不到isa的指向链的

    + (Class)class {
        return self;
    }
    
    - (Class)class {
        return object_getClass(self);
    }
    
    Class object_getClass(id obj) {
        if (obj) {
            return obj->getIsa();
        } else {
            return Nil;
        }
    }
    
  4. 元类(meta-class)之所以重要,是因为它储存着一个类的所有类方法,每个类都会有一个单独的meta-class,因为每个类的类方法基本不可能完全相同。

  5. 代码检验:

    Son *son = [[Son alloc] init];
    NSLog(@"instance: %p", son);
    NSLog(@"class: %p", object_getClass(son));
    NSLog(@"meta class: %p", object_getClass(object_getClass(son)));
    NSLog(@"root meta: %p", object_getClass(object_getClass(object_getClass(son))));
    NSLog(@"root meta's meta: %p", object_getClass(object_getClass(object_getClass(object_getClass(son)))));
    

    Log结果为:


    Screen-Shot-2018-01-21-at-6.36.24-PM-1.png

    可以验证isa的指向链是正确的


类族(class cluster)

  1. 先看一个NSNumber例子:

    NSNumber *integerNumber = [NSNumber numberWithInteger:1];
    NSLog(@"class: %p", object_getClass(integerNumber));
    NSLog(@"class: %p", [NSNumber class]);
    

    Log结果为:


    Screen Shot 2018-01-21 at 6.43.01 PM.png

    很明显,两个class返回的指针地址是不一样的。object_getClass(integerNumber)返回的应该是integerNumber isa对应的类NSNumber, [NSNumber class]返回的是其自己,也同样是NSNumber,按道理说这两个方法返回的指针应该是一样的啊。还有一个很奇怪的是如果是我们自己的Son类, 同样的两个方法返回的指针地址是一致的

  2. 原因其实很简单,[NSNumber numberWithInteger:1]返回的其实不是一个NSNumber类,是其的一个子类__NSCFNumber,为什么?因为NSNumber是一个类族,那为什么Son类返回的就一样的呢?因为Son不是一个类族

  3. 类族概念: 一个父类有好多子类,父类在返回自身对象的时候,向外界隐藏各种细节,根据不同的需要返回的其实是不同的子类对象,这其实就是抽象类工厂的实现思路

  4. 这样就解释了我之前遇到的swizzle NSMutableDictionarysetObject:forKey不生效的问题,我在NSMutableDictionary扩展中swizzle的是[self class]返回的类,相当于是[NSMutableDictionary class],这个方法返回的指针应该是NSMutableDictionary,但是我们项目里创建的NSMutableDictionary的实例其实对应的类是__NSDictionaryM,所以这就能解释为什么没有swizzle到我们项目里创建的NSMutableDictionary实例的setObject:forKey方法

转载请注明出处,原文地址:http://www.kobedai.me/objc-runtime_1/

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

推荐阅读更多精彩内容