美文网首页ios
图解isKindOfClass和isMemberOfClass

图解isKindOfClass和isMemberOfClass

作者: 叶孤城1993 | 来源:发表于2020-09-17 17:10 被阅读0次

    我们在开发中经常会用到 isKindOfClass: 来判断一个 obj 是不是某个类型。
    我们所有的知识点都基于“类”的isa

    isKindOfClass:

    查看objc4源码,我们会看到,无论是谁调用isKindOfClass: 都会进入objc_opt_isKindOfClass C函数。

    这个C函数位于NSObject.mm文件

    // Calls [obj isKindOfClass]
    // 当obj调用isKindOfClass时,objc_opt_isKindOfClass会被触发
    // obj是一个id类型,id是一个objc_object结构体指针,意味着,传进来的可以是时类,也可以是类的实例对象
    // otherClass就是isKindOfClass的参数,我们当初传进去的cls
    BOOL objc_opt_isKindOfClass(id obj, Class otherClass)
    {
    #if __OBJC2__
        if (slowpath(!obj)) return NO;
        
        Class cls = obj->getIsa();   // 此处的cls仅是obj的第一个isa
        if (fastpath(!cls->hasCustomCore())) {
            
            // otherClass 从obj的ISA开始,依次和ISA的父类比较,直到找到或者父类为nil结束
            // 当父类为nil意味着最后一个和otherClass比较的是NSObject根类。
            for (Class tcls = cls; tcls; tcls = tcls->superclass) {
                if (tcls == otherClass) return YES;
            }
            return NO;
        }
    #endif
        return ((BOOL(*)(id, SEL, Class))objc_msgSend)(obj, @selector(isKindOfClass:), otherClass);
    }
    

    得到一个信息:

    • 一切皆从调用者obj的isa开始,然后顺着superclass走下去,直到找到cls或superclass为nil结束
    • 当superclass为nil,意味着最后的根类NSObject也不是cls,返回flase。

    在isa走位图的基础上,分析不同情况时,cls的比较判断路径:

    调用 + (BOOL)isKinsOfClass:(Class)cls

    用isa走位图来表示,蓝色部分就是cls依次比较的链路。

    如果是类对象SubClass 调用isKindOfClass

    image

    文字描述:

    • 从类的ISA——元类开始。判断它不是cls
      • 如果是,返回true
    • 如果不是,继续用元类的superclass和cls比较,看是不是cls
    • 直到根类NSObject也比较完

    判断顺序: SubClass -> MetaClass->MetaClass->...->RootMetaClass->NSObject

    其中, 类对象调用方法的本质 是判断 cls 是不是 元类的继承链 上的任意一个

    如果是元类MetaClass 调用isKindOfClass

    还是上图, MetaClassISA 指向 RootMetaClass ,所以从 RootMetaClass 开始比较判断是不是我们传入的cls,如果不是,再看根类NSObject是不是,如果NSObject也不是,就彻底没有了,返回false,

    image
    cls判断顺序如图:MetaClass -> RootMetaClass->NSObject

    对象obj 调用- (BOOL)isKinsOfClass:(Class)cls

    从isa指向的类对象开始,判断是不是cls;如果不是,看类对象的父类,逐级判断是不是cls;直到找到返回true,或者判断到NSObject依然不是,返回false结束。

    image
    cls判断顺序如图所示:object -> SubClass -> SubClass ->...->NSObject

    本质是判断 cls 是不是 类继承链 上的任意一个

    isMemberofClass

    调用 +(BOOL)isMemberofClass:(Class)clss

    + (BOOL)isMemberOfClass:(Class)cls 底层源码

    + (BOOL)isMemberOfClass:(Class)cls {
        return self->ISA() == cls;
    }
    

    不用像isKindOfClass循环直到找到或nil,他只要比较cls是不是我当前的isa指向,是返回true,不是返回false。

    这个方法单纯地用来判断,cls是不是调用者的isa !!

    如果是类对象SubClass 调用isMemberofClass :传入任何类对象都是false

    因为类的isa不指向类对象 ,参见下图

    image

    如果是元类MetaClass 调用isMemberofClass:传入任何类对象也都是false

    1. 原因同上
    2. 元类的isa只指向根元类NSObejct
    image

    对象obj调用 -(BOOL)isMemberofClass:(Class)clss

    - (BOOL)isMemberOfClass:(Class)cls 底层源码:

    - (BOOL)isMemberOfClass:(Class)cls {
        return [self class] == cls;
    }
    - (Class)class {
        return object_getClass(self); // 获取当前的isa指向的类
    }
    
    

    只要判断对象的isa,也就是图中的SubClass是不是我们传入的cls


    image
    1. 只判断自己的类对象是不是传入的cls
    2. 只接受类对象传入 ,因为没有isa,不存在元类的介入

    测试用例验证输出:

    [图片上传失败...(image-338f4-1600333808410)]

    总结:

    当调用者是——类对象SubClass

    +(BOOL)isKindOfClass:(Class)cls :

    • 判断cls是不是 元类->父类的元类->父父类的元类->...->根元类->NSObject (元类的superclass继承链)其中一个。
    • cls 传除NSObject.class外的任意类对象均为false。

    +(BOOL)isMemeberOfClass:(Class)cls

    • 判断cls是不是自己的isa,

    当调用者是——元类MetaClass

    +(BOOL)isKindOfClass:(Class)cls

    • 判断cls是不是 根元类->NSObject 中的任意一个

    +(BOOL)isMemeberOfClass:(Class)cls

    • 判断cls是不是根元类

    当调用者是——对象Obj

    -(BOOL)isKindOfClass:(Class)cls

    • 判断cls是不是 **类对象->父类->...->NSObject **(superclass继承链)其中一个

    -(BOOL)isMemeberOfClass:(Class)cls

    • 判断 cls 是不是自己的 类(类对象)

    相关文章

      网友评论

        本文标题:图解isKindOfClass和isMemberOfClass

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