首先我们先来看一下这道面试题是啥?
题目看着非常简单,我是先创建了一个继承NSObject的GDPerson类;
GDPerson类的.h文件
GDPerson类的.m文件
再看一下我们viewController.m里面的代码:
作为一个开发者,有一个学习的氛围跟一个交流圈子特别重要,这是一个我的iOS开发交流群:130 595 548,不管你是小白还是大牛都欢迎入驻 ,让我们一起进步,共同发展!(群内会免费提供一些群主收藏的免费学习书籍资料以及整理好的几百道面试题和答案文档!)
这是题目
请问:
1.print能不能调用成功?如果不能会怎么样?如果能的话调用结果是什么?
这个又是一个更扯的面试题,真正开发的时候,谁也不会这么写,这个还是主要考你基础!相信你看到这个题目之后应该心中已经有了答案,要不知道结果,要么可能知道结果,要么犹豫不决,要么不知道哈哈!
其实这个面试题考点比较多,考点如下:
1.你要了解super的本质,第一个参数要传结构体
2.函数的堆栈分配问题
3.消息机制,调用方法是怎么调用
4.访问成员变量的本质
这样,我们先来看一下调用结果吧!
请看结果:
面试题运用结果
这里跟你想的答案一样吗?
这样我在cls前面加一段代码,我们再看一下结果:
面试题运用结果
首先我们立刻会有2个疑问:
1.为什么能调用成功?
2.为什么self.name调用结果是viewController?
一.为什么能调用成功?
[(__bridge id)obj print];由之前的学习,我们知道这个代码的本质是:给obj发一条print的消息,就会去通过obj的isa找到obj的类对象,去找方法列表,这个是非常清楚的.这个能调用成功,说明(__bridge id)obj也存在我们之前说的是isa指针这个东西
我画个图,这样理解的比较清楚.
cls是指向GDPerson,obj是指向cls,所以图如下:
上面绿色是[GDPerson class],图比较模糊
再请看下面的代码:
person指向GDPerson的实例变量,而GDPerson的实例变量是包括isa和成员变量等等,这个也很清楚.而isa是指向GDPerson的类对象,所以请看下面的图:
我们根据之前的源码分析知道,isa和_name是存在一个结构体,而对于结构体来说,第一个成员变量的地址值就是这个结构体的地址.所以person就是指向isa.
好了,这两个图我们分析清楚了以后,你看这两个图是不是很类似,几乎是一样的,我们再看下面的一个图:
上面绿色是[GDPerson class],图比较模糊
所以从上面的结构上,你看是不是就是一样的,obj就相当于person,所以能调用,这个比较抽象.说白了,你上面写的cls就是isa作用,因为我们知道指向类对象的指针就是isa.isa就是存储类对象的地址值.而你可能有疑问cls里面都没有_name怎么能算是GDPerson对象呢?注意,我们是调用print方法,调用方法没有说一定有_name成员变量!是这样吧!它是跟有没有成员变量是没有关系的.
第二个角度解释:obj怎么找到cls,就是通过cls对象的前8个字节的内存地址找到它,前8个字节也是结构体的地址,通过地址就能很容易找到class对象,是这样的.所以它能够调用成功!这就是调用的本质,后面那条线的调用也是如此.
二.为什么self.name调用结果是其他的?
首先你要知道堆栈排列的知识点,请看下图:
这些变量都是存在栈空间的,而且内存地址是由高地址到底地址.
好我们再看下面这个之前的图:
我们画一下这些地址排列如下
上面代码的结构示意图
上面绿色是[GDPerson class],图比较模糊,这个图很清楚.
现在我们来回忆一下:之前我们定义一个对象,比如上面的GDPerson这个类,它的底层生成的结构是下面这个样子的
structGDPerson_IMPL
{
Class isa;
NSString *_name;
}
上面这个结构体就是底层的结构,现在我们想一下,怎么找到_name这个地址,肯定是找到isa指针的地址加上8个字节就能找到_name吧?看下图
这个应该是很明显,找name就是通过isa找到name对应的这块内存地址就行了.
现在大家知道下结果了吧?上面的cls就是我们的isa指针,所以找name就找到了test这里面!好我们再运行一下看看
这里你定义test,你定义任何其他的都是一样,都会找到cls前面声明的变量.比如我再定义一个objct再看下.
输出的结果就是cls上面最近的一个创建的.还有一个未解决就是self.name调用结果是viewController
三、为什么self.name调用结果是viewController?
我们把test变量去掉,结果就会是viewController 我直接说了这个主要是[super viewDidLoad]影响;从上一张博客我们知道
super做了什么事它底层是这样实现的(上个博客说得很清楚): objc_msgSendSuper({ self,[UIViewController Class]} ,@selector(viewDidLoad));其他就是做了这件事@selector(viewDidLoad)也可以写出sel_registerName("viewDidLoad")
这个肯定要开始定义一个局部的结构体才能传入 objc_msgSendSuper这个方法.所以最高地址是abc这个结构体,而结构体的第一个参数的地址就是结构体的地址,所以输出的就是self也就是viewController.
如下图:
就会找到self
下面我们通过内存来证明一下这个东西:
这个题目涉及的知识点还是比较多,如果直接给你题目凭空想想,还是很难想出答案,好了,就说这么多了