先来看看一道常见的面试题:
void isKindOfAndisMemeberOf() {
BOOL b1 = [(id)[NSObject class] isKindOfClass:[NSObject class]];
BOOL b2 = [(id)[NSObject class] isMemberOfClass:[NSObject class]];
BOOL b3 = [(id)[Person class] isKindOfClass:[Person class]];
BOOL b4 = [(id)[Person class] isMemberOfClass:[Person class]];
BOOL b5 = [(id)[NSObject alloc] isKindOfClass:[NSObject class]];
BOOL b6 = [(id)[NSObject alloc] isMemberOfClass:[NSObject class]];
BOOL b7 = [(id)[Person alloc] isKindOfClass:[Person class]];
BOOL b8 = [(id)[Person alloc] isKindOfClass:[Person class]];
NSLog(@"\n%hhd\n%hhd\n%hhd\n%hhd\n%hhd\n%hhd\n%hhd\n%hhd\n",b1,b2,b3,b4,b5,b6,b7,b8);
}
要想知道输出结果为什么,必须先知道isKindOfClass与isMemberOfClass的具体实现。
+ (BOOL)isMemberOfClass:(Class)cls {
return self->ISA() == cls;
}
- (BOOL)isMemberOfClass:(Class)cls {
return [self class] == cls;
}
+ (BOOL)isKindOfClass:(Class)cls {
for (Class tcls = self->ISA(); tcls; tcls = tcls->superclass) {
if (tcls == cls) return YES;
}
return NO;
}
- (BOOL)isKindOfClass:(Class)cls {
for (Class tcls = [self class]; tcls; tcls = tcls->superclass) {
if (tcls == cls) return YES;
}
return NO;
}
解析
-
+ (BOOL)isKindOfClass:(Class)cls实现判断元类是不是为当前类,再判断元类的父类是不是当前类,依次判断下去。所以:
-
NSObject的元类为根元类不同于NSObject,根元类的父类为NSObject与NSObject相同,所以b1为true。 -
Person的元类为Person(元类)不同,Person(元类)的父类为根元类不同,根元类的父类为NSObject不同,NSObject父类为nil不同,所以b3为false。
-
+ (BOOL)isMemberOfClass:(Class)cls是去和元类做比较,元类与类不同,所以b2与b4为false。
3.- (BOOL)isKindOfClass:(Class)cls与- (BOOL)isMemberOfClass:(Class)cls,都是判断实例对象的类是不是对应的类,区别在于isKindOf会往父类找,所以b5到b8都为true。
运行结果如图
