作为iOS开发,这道面试题你能答出来,说明你基础很OK!

3,252 阅读6分钟

首先我们先来看一下这道面试题是啥?

​题目看着非常简单,我是先创建了一个继承NSObject的GDPerson类;

GDPerson类的.h文件

GDPerson类的.m文件

再看一下我们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,所以图如下:

上面绿色是[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

下面我们通过内存来证明一下这个东西:

这个题目涉及的知识点还是比较多,如果直接给你题目凭空想想,还是很难想出答案,好了,就说这么多了

接下来我会继续介绍runtime的实战应用,来继续学习runtime.

如果觉得我写得对您有所帮助,请关注我,我会持续更新😄