//.h文件的实现
@interface NSObject (Test)
- (void)test;
+ (void)test1;
@end
@interface ClassA : NSObject
@end
//.m文件的实现
@implementation NSObject (Test)
+ (void)test {
NSLog(@"aaa");
}
- (void)test1 {
NSLog(@"bbb");
}
@end
@implementation ClassA
@end
请问:
[NSObject test1]; //打印结果”bbb“
[ClassA test1];//打印结果”bbb“
[[[NSObject alloc] init] test];//崩溃
[[[ClassA alloc] init] test];//崩溃
图释
instance of root class就是我们所说的对象,
root class就是对象所属的类,
meta class就是类所属的元类,
对象,类,元类都是结构体,
其中包含isa指针,对象的isa指针指向的类,类的isa指针指向的元类,元类的isa指针指向的根元类(Root meta class)。
我们调用方法时,isa指针就会到对象所指类的方法列表中寻找,
比如:
- 1 实例对象调用方法,那么runtime就会去实例对象所指的类中去寻找该方法,找到则执行,找不到该方法,先去分类找,如果分类找不到,就会去该类的父类中去寻找,依次顺序寻找。
- 2 类调用方法,runtime会去元类中去寻找该方法,如果找不到该方法,就会去该元类的父类中寻找。
[NSObject test1]此题
首先,我为NSObject添加了一个分类,分类中有声明+ (void)test1类方法,但没有实现该类方法,却实现了另一个- (void)test1实例方法!!!
当NSObject调用类方法时,按照上图所示,runtime就会在NSObject的meta class中寻找方法,但是没找到,那么按照上述,就会去元类的父类方法中寻找,由于NSObject的父类是NSObject类,NSObject类实现了test1的实例方法,由于runtime寻找方法是根据方法名去寻找(注意:只有实现的方法才能被runtime找到,只声明的方法不能被找到),如果类中的实例方法与声明的类方法名称相同,那么就会被runtime找到,找到后就runtime就会触发这个方法,所以 [NSObject test1]执行结果是 打印"bbb";
举一反三
如果我们不是为NSObject添加分类,而是为另一个非基类的类添加分类,会出现这种问题吗?
假设我们为NSArray添加该分类,我们同样按照图中所示一步一步看,我们调用类方法,那么runtime就会在NSArray的meta class中去寻找该方法,但是没找到,那么就会去它的父类的meta class中寻找,NSArray的父类是NSObject,很明显,NSObject meta class中也没有这个方法,那就继续往NSObject meta class的父类找,但是按图中所示,NSObject meta class的父类就是NSObject类了,很明显,NSObject类中也没有这个方法,那么就会返回nil,如果不做拦截处理,程序自然就crash了。所以这种情况只会出现在基类上。