案例如下 : 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]
的流程 :
- 在NSObject元类的方法列表中中查找
foo
方法.如果没找到,转向NSObject元类的父类中查找 - 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
才会出现,在其他类如果有这种情况,必定会产生崩溃