先来看看一道常见的面试题:
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
。