圖解isKindOfClass和isMemberOfClass方法

咱們在開發中常常會用到 isKindOfClass: 來判斷一個 obj 是否是某個類型。 咱們全部的知識點都基於「類」的isamarkdown

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);
}
複製代碼

獲得一個信息:ui

  • 一切皆從調用者obj的isa開始,而後順着superclass走下去,直到找到cls或superclass爲nil結束
  • 當superclass爲nil,意味着最後的根類NSObject也不是cls,返回flase。

在isa走位圖的基礎上,分析不一樣狀況時,cls的比較判斷路徑:spa

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

用isa走位圖來表示,藍色部分就是cls依次比較的鏈路。3d

若是是類對象SubClass 調用isKindOfClass

文字描述:指針

  • 從類的ISA——元類開始。判斷它不是cls
    • 若是是,返回true
  • 若是不是,繼續用元類的superclass和cls比較,看是否是cls
  • 直到根類NSObject也比較完

判斷順序: SubClass -> MetaClass->MetaClass->...->RootMetaClass->NSObject code

其中, 類對象調用方法的本質 是判斷 cls 是否是 元類的繼承鏈  上的任意一個orm

若是是元類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

  1. 緣由同上 
  2. 元類的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

  1. 只判斷本身的類對象是否是傳入的cls 
  2. 只接受類對象傳入 ,由於沒有isa,不存在元類的介入

測試用例驗證輸出:

image.png

總結:

當調用者是——類對象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 是否是本身的 類(類對象) 
相關文章
相關標籤/搜索