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

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

作者: iOS的火影乱斗 | 来源:发表于2021-04-06 13:43 被阅读0次

    作者:GDCoder
    链接:https://juejin.cn/post/6946507569535909919

    前言:最近应该有很多小伙伴去跳槽面试的吧,相信各位有的已经顺利收到offer了,而有些则是碰壁了,那么我在这里给大家准备了相关面试资料,还有相关算法资料。想了解的可找我拿

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

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

    GDPerson类的.h文件


    GDPerson类的.m文件

    再看一下我们viewController.m里面的代码:

    这是题目

    请问:

    1.print能不能调用成功?如果不能会怎么样?如果能的话调用结果是什么?

    这个又是一个更扯的面试题,真正开发的时候,谁也不会这么写,这个还是主要考你基础!相信你看到这个题目之后应该心中已经有了答案,要不知道结果,要么可能知道结果,要么犹豫不决,要么不知道哈哈!

    其实这个面试题考点比较多,考点如下:

    1.你要了解super的本质,第一个参数要传结构体

    2.函数的堆栈分配问题

    3.消息机制,调用方法是怎么调用

    4.访问成员变量的本质

    这样,我们先来看一下调用结果吧!

    请看结果:

    面试题运用结果

    这里跟你想的答案一样吗?

    这样我在cls前面加一段代码,我们再看一下结果:

    作为一个开发者,有一个学习的氛围跟一个交流圈子特别重要,这是一个我的iOS交流圈子:[891 488 181] ,不管你是小白还是大牛欢迎入驻 ,分享BAT,阿里面试题、面试经验,讨论技术, 大家一起交流学习成长!

    面试题运用结果

    首先我们立刻会有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

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

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

    文章到这里就结束了,你也可以私信我及时获取最新资料以及面试相关资料。如果你有什么意见和建议欢迎给我留言。

    相关文章

      网友评论

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

        本文链接:https://www.haomeiwen.com/subject/hpmskltx.html