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

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

#import 

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才会出现,在其他类如果有这种情况,必定会产生崩溃

你可能感兴趣的:(Runtime面试题-[NSObject foo]和[[NSObject new] foo])