白话文之面试题 isKindOfClass & isMemberOfClass

既然你能看到这想必也是应该一时间脑子没转明白是怎么回事,
如果你一下就明白怎么回事了你也就不会点到这边文章里了
好的 那就以大白话的形式 讲清楚 "这玩应"他是怎么来的, 又是怎么没的"

先来看一道面试题

不要用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 指针来 判断 是不是 参数传入的类或者其派生类(父类)

  1. [NSObject class] -> isa 确实 不是NSObject
  2. 往父元类上倒,父元类找不到 就继续往根元类上倒, 最后会倒到根元类的的父类
  3. NSObject根源类的父类是谁呢? 是NSObject自己!
  4. isKindOfClass 条件成立 结果为 true

现在你倒明白了么?

如果还有点蒙 来看张经典的不能再经典的图

白话文之面试题 isKindOfClass & isMemberOfClass_第1张图片
image.png

现在呢?

最后在回到上面的面试题

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])];
使用 runtimeobject_getClass 这个函数就可以了

既然能看到这里就证明你还是有想法在编程的领域不断深入的,现实的环境很残酷,但是不要放弃,加油~~

-----技术本没错,对错在于拥有技术的人的心

你可能感兴趣的:(白话文之面试题 isKindOfClass & isMemberOfClass)