Runtime面试题-[NSObject foo]和[[NSObject new] foo]

案例如下 : NSObject的Category中定义了一个实例方法和一个类方法,但是在.m文件中只有实例方法的实现

#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface NSObject (Simple)

- (void)foo;
+ (void)foo;

@end

NS_ASSUME_NONNULL_END


#import "NSObject+Simple.h"

@implementation NSObject (Simple)


- (void)foo
{
    NSLog(@"instance method");
}

@end

执行以下代码:

[[NSObject new] foo];
[NSObject foo];

//打印结果如下:
2019-06-25 16:29:42.888526+0800 Runtime_Simple[68493:2251741] instance method
2019-06-25 16:29:42.888681+0800 Runtime_Simple[68493:2251741] instance method

在弄明白这个案例之前,首先要搞清楚OC中类继承关系和消息发送转发流程,我之前做过一些相关解释译文 : 什么是Objective-C中的元类Runtime-消息发送与转发

下面来看下[NSObject foo]的流程 :

  1. 在NSObject元类的方法列表中中查找foo方法.如果没找到,转向NSObject元类的父类中查找
  2. NSObject元类的父类是NSObject类,找到foo方法的实现,执行该方法

问题 : 到这里有个问题,类方法和实例方法有啥区别,为什么执行的是类方法,找到实例方法的实现时也会执行?其实上面两篇文章已经能解释这个问题,下面来解释下.

首先我们clang一下上面的方法调用 :

((void (*)(id, SEL))(void *)objc_msgSend)((id)((NSObject *(*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("NSObject"), sel_registerName("new")), sel_registerName("foo"));
((void (*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("NSObject"), sel_registerName("foo"));

可以看到,clang之后,识别两个方法都是通过sel_registerName("foo")返回的SEL指针,没有区别,所以OC中实例方法和类方法知识存储位置不同,一个在类对象中,一个在类的元类中.所以在 NSObject元类的父类即NSObject类对象中同样可以匹配到实例方法foo.

这种情况只会在NSObject才会出现,在其他类如果有这种情况,必定会产生崩溃

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,161评论 1 32
  • 前言:面试笔试都是必考语法知识点。请认真复习和深入研究OC。 目录:iOS-面试题-OC基础篇 (1) - (84...
    麦穗0615阅读 4,289评论 0 33
  • 出题者简介: 孙源(sunnyxx),目前就职于百度,负责百度知道 iOS 客户端的开发工作,对技术喜欢刨根问底和...
    戈多_于勒阅读 1,821评论 0 5
  • 杏花仙子 头顶一片云霞, 身披一袭轻纱 在御花园里轻轻落下 纤纤玉手捧出一幅幅—— 承载着生命的画 粉嫩的唇 娇美...
    春韵留芳阅读 277评论 0 0
  • 室友:“如果有人说我爱你 怎么回答才会赢?” 我:“反弹。” 室友:“哈哈哈。 那是平...
    胡雨洁2277阅读 258评论 0 1