我们在开发中经常会用到 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
文字描述:
- 从类的ISA——元类开始。判断它不是cls
- 如果是,返回true
- 如果不是,继续用元类的superclass和cls比较,看是不是cls
- 直到根类NSObject也比较完
判断顺序: SubClass -> MetaClass->MetaClass->...->RootMetaClass->NSObject
其中, 类对象调用方法的本质 是判断 cls
是不是 元类的继承链
上的任意一个
如果是元类MetaClass
调用isKindOfClass
还是上图, MetaClass
的 ISA
指向 RootMetaClass
,所以从 RootMetaClass
开始比较判断是不是我们传入的cls,如果不是,再看根类NSObject是不是,如果NSObject也不是,就彻底没有了,返回false,
cls判断顺序如图:MetaClass ->
RootMetaClass->NSObject
当对象obj
调用- (BOOL)isKinsOfClass:(Class)cls
从isa指向的类对象开始,判断是不是cls;如果不是,看类对象的父类,逐级判断是不是cls;直到找到返回true,或者判断到NSObject依然不是,返回false结束。
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不指向类对象 ,参见下图
如果是元类MetaClass
调用isMemberofClass:传入任何类对象也都是false
- 原因同上
- 元类的isa只指向根元类NSObejct
当对象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
- 只判断自己的类对象是不是传入的cls
- 只接受类对象传入 ,因为没有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
是不是自己的 类(类对象)