美文网首页iOS面试iOS面试基础知识点
NSObject调用分类的类方法的坑

NSObject调用分类的类方法的坑

作者: 爱笑的眼睛super | 来源:发表于2019-08-28 18:09 被阅读0次
    //.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基类尽量不要为其添加同名的实例和类方法,避免出现这种调用类方法,触发实例方法的情况。如果要同名,请保证俩个方法都必须实现。

    相关文章

      网友评论

        本文标题:NSObject调用分类的类方法的坑

        本文链接:https://www.haomeiwen.com/subject/xhnrectx.html