iOS面试iOS面试基础知识点

NSObject调用分类的类方法的坑

2019-08-28  本文已影响0人  爱笑的眼睛super
//.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指针就会到对象所指类的方法列表中寻找,
比如:

[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基类尽量不要为其添加同名的实例和类方法,避免出现这种调用类方法,触发实例方法的情况。如果要同名,请保证俩个方法都必须实现。

上一篇 下一篇

猜你喜欢

热点阅读