NSObject调用分类的类方法的坑

//.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];//崩溃
1.jpg
图释

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了。所以这种情况只会出现在基类上。

结论:OC基类尽量不要为其添加同名的实例和类方法,避免出现这种调用类方法,触发实例方法的情况。如果要同名,请保证俩个方法都必须实现。

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

推荐阅读更多精彩内容