首先我们先来看一下这道面试题是啥?
题目看着非常简单,我是先创建了一个继承NSObject的GDPerson类;
再看一下我们viewController.m里面的代码:
请问:
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,所以图如下:
再请看下面的代码:
person指向GDPerson的实例变量,而GDPerson的实例变量是包括isa和成员变量等等,这个也很清楚.而isa是指向GDPerson的类对象,所以请看下面的图:
我们根据之前的源码分析知道,isa和_name是存在一个结构体,而对于结构体来说,第一个成员变量的地址值就是这个结构体的地址.所以person就是指向isa.
好了,这两个图我们分析清楚了以后,你看这两个图是不是很类似,几乎是一样的,我们再看下面的一个图:
所以从上面的结构上,你看是不是就是一样的,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
下面我们通过内存来证明一下这个东西:
这个题目涉及的知识点还是比较多,如果直接给你题目凭空想想,还是很难想出答案,好了,就说这么多了