有一道经典面试题关于isKindOfClass和isMemberOfClass
代码:
BOOL re1 = [(id)[NSObject class] isKindOfClass:[NSObject class]]; // 1
BOOL re2 = [(id)[NSObject class] isMemberOfClass:[NSObject class]]; // 0
BOOL re3 = [(id)[LGPerson class] isKindOfClass:[LGPerson class]]; // 0
BOOL re4 = [(id)[LGPerson class] isMemberOfClass:[LGPerson class]]; // 0
NSLog(@" re1 :%hhd\n re2 :%hhd\n re3 :%hhd\n re4 :%hhd\n",re1,re2,re3,re4);
BOOL re5 = [(id)[NSObject alloc] isKindOfClass:[NSObject class]]; // 1
BOOL re6 = [(id)[NSObject alloc] isMemberOfClass:[NSObject class]]; // 1
BOOL re7 = [(id)[LGPerson alloc] isKindOfClass:[LGPerson class]]; // 1
BOOL re8 = [(id)[LGPerson alloc] isMemberOfClass:[LGPerson class]]; // 1
NSLog(@" re5 :%hhd\n re6 :%hhd\n re7 :%hhd\n re8 :%hhd\n",re5,re6,re7,re8);
输出结果:
2019-12-31 14:49:22.734091+0800 LGTest[35237:2807868]
re1 :1
re2 :0
re3 :0
re4 :0
2019-12-31 14:49:22.735580+0800 LGTest[35237:2807868]
re5 :1
re6 :1
re7 :1
re8 :1
为什么结果是这样呢?
先放一个isa的指针图:
isa流程图.png
- 1、首先我们先看一下类的class方法
+ (Class)class {
return self;
}
- 2、看一下类的isKindOfClass的实现
Class object_getClass(id obj)
{
if (obj) return obj->getIsa();
else return Nil;
}
+ (BOOL)isKindOfClass:(Class)cls {
//
for (Class tcls = object_getClass((id)self); tcls; tcls = tcls->superclass) {
if (tcls == cls) return YES;
}
return NO;
}
我们发现isKindOfClass是循环不断获取self的isa指针以及父类的isa指针指向和cls做对比,通过上面isa的指向图,我们可以拿到下面代码逻辑:
BOOL re1 = [(id)[NSObject class] isKindOfClass:[NSObject class]]; // 1
先拿到NSObject的元类跟NSObject比,因为NSObject元类的父类是NSObject,所以结果是YES
BOOL re3 = [(id)[LGPerson class] isKindOfClass:[LGPerson class]]; // 0
而遍历LGPerson的父类的isa却是拿LGPerson的元类、NSObject的元类、NSObject和LGPerson进行对比,所以,结果是NO
- 3、再看一下类的isMemberOfClass的实现
+ (BOOL)isMemberOfClass:(Class)cls {
return object_getClass((id)self) == cls;
}
我们发现isMemberOfClass仅仅是拿到当前self的isa指针指向和cls对比,然后我们分析测试代码逻辑:
BOOL re2 = [(id)[NSObject class] isMemberOfClass:[NSObject class]]; // 0
NSObject的元类和NSObject匹配,不成立
BOOL re4 = [(id)[LGPerson class] isMemberOfClass:[LGPerson class]]; // 0
LGPerson的元类和LGPerson匹配,不成立
- 4、再看一下实例的isKindOfClass方法和isMemberOfClass的实现
- (BOOL)isKindOfClass:(Class)cls {
// 类 - NSObject 类 vs 父类 nil
for (Class tcls = [self class]; tcls; tcls = tcls->superclass) {
if (tcls == cls) return YES;
}
return NO;
}
- (BOOL)isMemberOfClass:(Class)cls {
return [self class] == cls;
}
我们可以发现对于对象方法,只是拿到对象的类和cls对比,所以下面四个结果都是YES。
网友评论