Runtime之Super探究

super方法调用如果转为C++代码可以看到被转成了:

objc_msgSendSuper()方法调用。

该方法需要两个参数:

//第一个参数:是一个objc_super的结构体。该结构体有两个成员。
struct objc_super {
  id receiver; //消息接受者
  Class cls;    // 消息接受者的父类,方法查找从该类开始。
}

//第二个参数:是调用的方法
@selector(method_name);

可以看到super是通过objc_msgSendSuper发送消息,其中消息接收者还是self。然后要发送的消息是从父类中开始查找,而不是跟往常一样从当前对象查找。

例如:

@implementation GLStudent
- (void)run {
    [super run];
    NSLog(@"GLStudent run");
}
@end

转化为C++代码:

struct __rw_objc_super arg = {
        self,//消息接收者
        class_getSuperclass(objc_getClass("GLStudent"))
    };
    
objc_msgSendSuper(
                  arg,
                  sel_registerName("run")
                   );

有些时候转换的C++代码只能作为一个参考,真正的底层调用还要从汇编的角度分析。

从汇编角度看,super方法调用被转换成了objc_msgSendSuper2(),objc_msgSendSuper2()由汇编实现。描述是这样的:

id objc_msgSendSuper2(struct objc_super2 *super, SEL op, ...)
  
struct objc_super2 {
    id receiver;  // 消息接收者
    Class current_class;  // receiverClass(消息接收者的class对象)
};

尽管第一个参数发生了变化,其实无非实在objc_msgSendSuper2() 内部,根据current_class获取了superClass。之后的逻辑与objc_msgSendSuper()是一样的

[self class] 与 [super class]

关于Runtime有一道很经典的题目:

  • 打印结果是什么呢?

    @interface GLPerson : NSObject
    @end
      
    @interface GLStudent : GLPerson
    @end
    
    - (instancetype)init
    {
        self = [super init];
        if (self) {
            NSLog(@"[self class] = %@",[self class]); //GLStudent
            NSLog(@"[self superclass] = %@",[self superclass]); //GLPerson
    
            NSLog(@"[super class] = %@",[super class]); //GLStudent
            NSLog(@"[super superclass] = %@",[super superclass]); //GLPerson
        }
        return self;
    }
    

[super class]输出的是GLStudent

[super superClass]输出的是GLPerson

这两个方法的输出与self调用输出无异。根据Super的底层原理可知:

  • 虽然使用super调用,但消息接受者仍然是self,只不过方法查找是在父类中开始查找。
  • class方法与superClass方法,都是NSObject中实现的。所以无论调用者是谁,最终都是在NSObject中找到的方法。

所以[self class][super class]调用结果都是一样的,[self superClass][super superClass]也同理。

isKindOfClass & isMemberOfClass

isKindOfClassisMemberOfClass是两个常用的方法,又因为语义相近常常容易混淆。

+ (Class)class {
    return self;
}

- (Class)class {
    return object_getClass(self);
}
  • isKindOfClass:

    + (BOOL)isKindOfClass:(Class)cls {
        for (Class tcls = object_getClass((id)self); tcls; tcls = tcls->superclass) {
            if (tcls == cls) return YES;
        }
        return NO;
    }
    
    - (BOOL)isKindOfClass:(Class)cls {
        for (Class tcls = [self class]; tcls; tcls = tcls->superclass) {
            if (tcls == cls) return YES;
        }
        return NO;
    }
    

    只要能通过该对象superclass链能找到cls,就会返回YES。反之,则返回NO

  • isMemberOfClass

    + (BOOL)isMemberOfClass:(Class)cls {
        return object_getClass((id)self) == cls;
    }
    
    - (BOOL)isMemberOfClass:(Class)cls {
        return [self class] == cls;
    }
    

    直接作比较,看是否相等。

    isKindOfClassisMemberOfClass的类方法中都用到了object_getClass()

    Class object_getClass(id obj) 
    {
      if (obj) return obj->getIsa();
      else return nil;
    }
    

    从源码可以看出,object_getClass()获取的是obj的isa指针。实例对象获取的是类对象,类对象获取的是元类对象。

isa_走位原理图.png

示例1

/*
1.[NSObject class]获取了基类对象,+isKindOfClass方法中,匹配时会获取[NSObject class]的isa指针获取NSObject的元类对象
2. NSObject的元类对象的superclass指针又指向[NSObjcet class],故返回YES.
*/
BOOL ret1 = [[NSObject class] isKindOfClass:[NSObject class]]; //YES
/*
最终比较的是,NSObject的类对象和NSObjcet的元类对象是否相等,故返回NO
*/
BOOL ret2 = [[NSObject class] isMemberOfClass:[NSObject class]]; //NO
BOOL ret3 = [[GLPerson class] isKindOfClass:[GLPerson class]]; //NO
BOOL ret4 = [[GLPerson class] isMemberOfClass:[GLPerson class]]; //NO

示例2

super_面试题.png

可以运行,运行的结果类似于:

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

推荐阅读更多精彩内容