Runtime之Super探究

2021-07-16  本文已影响0人  飞翔的船

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有一道很经典的题目:

[super class]输出的是GLStudent

[super superClass]输出的是GLPerson

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

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

isKindOfClass & isMemberOfClass

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

+ (Class)class {
    return self;
}

- (Class)class {
    return object_getClass(self);
}
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
上一篇 下一篇

猜你喜欢

热点阅读