iOS中的 isKindOfClass 和 isMemberOfClass

先來(lái)看一個(gè)爛大街的面試題:

下面代碼結(jié)果如何?

BOOL res1 = [(id)[NSObject class] isKindOfClass:[NSObject class]];
BOOL res2 = [(id)[NSObject class] isMemberOfClass:[NSObject class]];
BOOL res3 = [(id)[TestObject class] isKindOfClass:[TestObject class]];
BOOL res4 = [(id)[TestObject class] isMemberOfClass:[TestObject class]];

答案:除了第一個(gè)是YES,其他三個(gè)都是NO。

在推測(cè)結(jié)果之前,首先要明白兩個(gè)問(wèn)題。isKindOfClassisMemberOfClass 的區(qū)別是什么?

isKindOfClass:
returns YES if the receiver is an instance of the specified class or an instance of any class that inherits from the specified class.
方法調(diào)用者是傳入的類(lèi)的實(shí)例對(duì)象,或者調(diào)用者是傳入類(lèi)的繼承者鏈中的類(lèi)的實(shí)例對(duì)象,則返回YES。

isMemberOfClass:
returns YES if the receiver is an instance of the specified class.
方法調(diào)用者必須是傳入的類(lèi)的實(shí)例對(duì)象才返回YES。

我們從Runtime源碼的角度來(lái)分析一下結(jié)果。

+ (BOOL)isMemberOfClass:(Class)cls {
    return object_getClass((id)self) == cls;
}

- (BOOL)isMemberOfClass:(Class)cls {
    return [self class] == cls;
}

+ (BOOL)isKindOfClass:(Class)cls {
    for (Class tcls = object_getClass((id)self); tcls; tcls = tcls->super_class) {
        if(tcls == cls) return YES;
    }
    return NO;
}
-(BOOL)isKindOfClass:(Class)cls {
    for(Class tcls = [self class]; tcls; tcls = tcls->super_class) {
        if(tcls == cls) return YES;
    }
    return NO;
}

平時(shí)開(kāi)發(fā)過(guò)程中只會(huì)接觸到對(duì)象方法的isKindOfClassisMemberOfClass,但是在NSObject類(lèi)中還隱式的實(shí)現(xiàn)了類(lèi)方法版本。不只這兩個(gè)方法,其他NSObject中的對(duì)象方法,都有其對(duì)應(yīng)的類(lèi)方法版本。因?yàn)樵贠C中,類(lèi)和元類(lèi)也都是對(duì)象。這四個(gè)調(diào)用由于都是類(lèi)對(duì)象發(fā)起調(diào)用的,所以最終執(zhí)行的都是類(lèi)方法版本。

先把Runtime的對(duì)象模型拿出來(lái),方便后面的分析。

因?yàn)檎{(diào)用類(lèi)的class方法,會(huì)把類(lèi)自身直接返回,所以第一次調(diào)用者是NSObject類(lèi)對(duì)象,+ (BOOL)isKindOfClass:(Class)cls方法的參數(shù)cls也是NSObject類(lèi)對(duì)象。

進(jìn)入到for循環(huán)中,在此說(shuō)明一下object_getClass()方法相當(dāng)于取出isa指針指向的類(lèi),所以會(huì)從NSObject的元類(lèi)開(kāi)始遍歷,所以第一次NSObject meta class != NSObject class,匹配失敗。第二次循環(huán)將tcls 設(shè)置為superclassNSObject class,NSObject class==NSObject class,匹配成功。

NSObject 能匹配成功,是因?yàn)檫@個(gè)類(lèi)比較特殊,在第二次獲取 superClass的時(shí)候,NSObject 元類(lèi)的superClass就是NSObject的類(lèi)對(duì)象,所以會(huì)匹配成功。而其他三種匹配,則都會(huì)失敗,各位同學(xué)可以去自己分析一下剩下三種。

參考資料

博客地址 作者:劉小壯

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容