图解isKindOfClass和isMemberOfClass

我们在开发中经常会用到 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 是不是自己的 类(类对象)
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。