既然你能看到这想必也是应该一时间脑子没转明白是怎么回事,
如果你一下就明白怎么回事了你也就不会点到这边文章里了
好的 那就以大白话的形式 讲清楚 "这玩应"他是怎么来的, 又是怎么没的"
先来看一道面试题
不要用Xcode去实操,就用眼睛看
请问这四句打印 都会是什么结果?
BOOL rez1 = [(id)[NSObject class] isKindOfClass:[NSObject class]];
BOOL rez2 = [(id)[NSObject class] isMemberOfClass:[NSObject class]];
BOOL rez3 = [(id)[UIView class] isKindOfClass:[UIView class]];
BOOL rez4 = [(id)[UIView class] isMemberOfClass:[UIView class]];
答案是 rez1 = true;
rez2,rez3,rez4均都是false;
不管你在网上看到的这题 还是进了这篇文章才看到的这题,就问你是不是懵逼了?
isKindOfClass & isMemberOfClass 我相信有点实际开发经验的同学 都能说出个123来, 而实际开发中也都用的没错,为啥到这里就发现 答案和自己脑子中想的不一样啊?
还是以打比方的形式来说明一下 坑在哪
比如 1+1 在什么情况下等于 3? (经不经典 意不意外?)
你不能说在 算错的情况下吧? 你也不能说在什么情况下都不等于3吧?
1(男人) + 1(女人) = 3(男人, 女人, 宝宝)
这才是正解, 原因不在于你回答的不对,而是你压根就没想到这方面去(换句现在话: 卧槽?! 还能这么玩?)
Binggo~! 上面这道题 就是这么玩的
Why?
@protocol NSObject
- (BOOL)isKindOfClass:(Class)aClass;
- (BOOL)isMemberOfClass:(Class)aClass;
@end
原因刨根的同学会点进头文件里显示的就这两句 没错吧?
好了你掉入陷阱了
- (BOOL)isKindOfClass:(Class)aClass;
问题就出在最前面的这个 " - " 减号上
都知道 减号是 对象方法 供类的实例来调用的, 那么请翻到文章的开头的那到面试题上 仔细瞧瞧 题目 是 一个实例对象在调用 对象方法么?
啥感觉?
卧槽?! 还能这么玩?
是不是1 + 1 = 3了呢?
其实在NSObject中 还有 isKindOfClass & isMemberOfClass 对应的类方法
为什么你被坑了? 因为这俩是隐藏的 你点到头文件里 看不到而已
@protocol NSObject
+ (BOOL)isKindOfClass:(Class)aClass;
+ (BOOL)isMemberOfClass:(Class)aClass;
@end
来来来,我们来回一下 isKindOfClass 这个方法的工作原理
// 对象方法 都做了些什么?
// 根据 obj->isa 指针来 判断 是不是 NSObject 这个类或者其派生类(父类)
是不是?
NSObject *obj = [[NSObject alloc] init];
BOOL rez1 = [obj isKindOfClass:[NSObject class]];
实例对象的 isa 指向的 是其类对象
obj->isa 指向的是 NSObject, 所以条件判断成立 结果为 true
这才是实际开发中 你能遇到的情况
而[(id)[UIView class] isKindOfClass:[UIView class]]
呢?
// 根据 [UIView class] -> isa 来看看 是不是 UIView这个类 或其派生类
是不是这样的?
[(id)[UIView class] isKindOfClass:[UIView class]];
那类对象的 isa 指向的是谁呢?
当然是指向 其 <<元类对象>> 啊
[UIView class] -> isa 应该叫 UIMetaView 吧? 我瞎编的 其实单纯看的话根本不知道叫啥
所以结果自然就是 false 了
这时候 如果你脑子还没转过来的话 可能会问 那 rez1 那个为啥等于 true 呢?
BOOL rez1 = [(id)[NSObject class] isKindOfClass:[NSObject class]];
结果为 true
来说说流程
isKindOfClass
会做些什么
会根据 ->isa 指针来 判断 是不是 参数传入的类或者其派生类(父类)
- [NSObject class] -> isa 确实 不是NSObject
- 往父元类上倒,父元类找不到 就继续往根元类上倒, 最后会倒到根元类的的父类
- NSObject根源类的父类是谁呢? 是NSObject自己!
-
isKindOfClass
条件成立 结果为 true
现在你倒明白了么?
如果还有点蒙 来看张经典的不能再经典的图
现在呢?
最后在回到上面的面试题
BOOL rez1 = [(id)[NSObject class] isKindOfClass:[NSObject class]];
BOOL rez2 = [(id)[NSObject class] isMemberOfClass:[NSObject class]];
BOOL rez3 = [(id)[UIView class] isKindOfClass:[UIView class]];
BOOL rez4 = [(id)[UIView class] isMemberOfClass:[UIView class]];
既然下面三个 结果都是 false 那怎样才能让结果变为 true 呢?
BOOL rez3 = [(id)[UIView class] isKindOfClass: object_getClass([UIView class])];
使用runtime
的object_getClass
这个函数就可以了