先來(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)題。isKindOfClass 和 isMemberOfClass 的區(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ì)象方法的isKindOfClass和isMemberOfClass,但是在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è)置為superclass的NSObject class,NSObject class==NSObject class,匹配成功。
NSObject 能匹配成功,是因?yàn)檫@個(gè)類(lèi)比較特殊,在第二次獲取 superClass的時(shí)候,NSObject 元類(lèi)的superClass就是NSObject的類(lèi)對(duì)象,所以會(huì)匹配成功。而其他三種匹配,則都會(huì)失敗,各位同學(xué)可以去自己分析一下剩下三種。