美文网首页iOS面试
iOS Runtime面试题(代码题 一)

iOS Runtime面试题(代码题 一)

作者: 程序员_秃头怪 | 来源:发表于2019-08-03 13:40 被阅读3次

    题目一:下面的代码输出什么?

    @implementation Son : Father
    - (id)init {
        self = [super init];
        if (self) {
            NSLog(@"%@", NSStringFromClass([self class]));
            NSLog(@"%@", NSStringFromClass([super class]));
        }
        return self;
    }
    @end
    

    结果: Son / Son

    分析:

    对于上面的答案,第一个的结果应该是我们的预期结果,但是第二个结果却让我们很费解了。

    那我们利用前面文章讲过的知识点来分析一下整个的流程。

    因为,Son 及 Father 都没有实现 -(Class)calss 方法,所以这里所有的调用最终都会找到基类 NSObject 中,并且在其中找到 -(Class)calss 方法。那我们需要了解的就是在 NSObject 中这个方法的实现了。

    在 NSObject.mm 中可以找到 -(Class)class 的实现:

    - (Class)class {
        return object_getClass(self);
    }
    

    在 objc_class.mm 中找到 object_getClass 的实现:

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

    ps:上面的方法定义可以去官方OpenSource中下载源码哦。

    可以看到,最终这个方法返回的是,调用这个方法的 objc 的 isa 指针。那我们只需要知道在题干中的代码里面最终是谁在调用 -(Class)class 方法就可以找到答案了。

    接下来,我们利用 clang -rewrite-objc 命令,将题干的代码转化为如下代码:

    NSLog((NSString *)&__NSConstantStringImpl__var_folders_8k_cgm28r0d0bz94xnnrr606rf40000gn_T_Car_3f2069_mi_0, NSStringFromClass(((Class (*)(id, SEL))(void *)objc_msgSend)((id)self, sel_registerName("class"))));
    NSLog((NSString *)&__NSConstantStringImpl__var_folders_8k_cgm28r0d0bz94xnnrr606rf40000gn_T_Car_3f2069_mi_1, NSStringFromClass(((Class (*)(__rw_objc_super *, SEL))(void *)objc_msgSendSuper)((__rw_objc_super){(id)self, (id)class_getSuperclass(objc_getClass("Car"))}, sel_registerName("class"))));
    

    从上方可以得出,调用 [Father class] 的时候,本质是在调用

    objc_msgSendSuper(struct objc_super *super, SEL op, ...)
    

    struct objc_super 的定义如下:

    struct objc_super {
        /// Specifies an instance of a class.
        __unsafe_unretained _Nonnull id receiver;
    
        /// Specifies the particular superclass of the instance to message. 
    #if !defined(__cplusplus)  &&  !__OBJC2__
        /* For compatibility with old objc-runtime.h header */
        __unsafe_unretained _Nonnull Class class;
    #else
        __unsafe_unretained _Nonnull Class super_class;
    #endif
        /* super_class is the first class to search */
    };
    

    从定义可以得知:当利用 super 调用方法时,只要编译器看到super这个标志,就会让当前对象去调用父类方法,本质还是当前对象在调用,是去父类找实现,super 仅仅是一个编译指示器。但是消息的接收者 receiver 依然是self。最终在 NSObject 获取 isa 指针的时候,获取到的依旧是 self 的 isa,所以,我们得到的结果是:Son。

    扩展一下: 看看下方的代码会输出什么?

    @interface Father : NSObject
    @end
    
    @implementation Father
    
    - (Class)class {
        return [Father class];
    }
    
    @end
    
    ---
    
    @interface Son : Father
    @end
    
    @implementation Son
    
    - (id)init {
        self = [super init];
        if (self) {
            NSLog(@"%@", NSStringFromClass([self class]));
            NSLog(@"%@", NSStringFromClass([super class]));
        }
        return self;
    }
    
    @end
    
    int main(int argc, const char * argv[]) {
        Son *foo = [[Son alloc]init];
        return 0;
    }
    
    ---输出:---
    Father
    Father
    

    面试题持续整理更新中,需要拿到第一手大厂面试题及答案文档可以添加 iOS进阶学习交流群:551346706 !结实人脉、讨论技术你想要的这里都有!

    相关文章

      网友评论

        本文标题:iOS Runtime面试题(代码题 一)

        本文链接:https://www.haomeiwen.com/subject/borcdctx.html