美文网首页
Super关键字

Super关键字

作者: coder_feng | 来源:发表于2019-07-09 20:44 被阅读0次

    面试题1:

    上代码图:

    Student.m

    student

    从打印结果里面看到super class 返回的是Student,而[super superclass] 返回Person,是不是感觉有点奇怪?现在我们就通过重写run方法,调用[super run] 查看super的本质

    xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc Student.m 然后进入Student.cpp 文件里面找到run方法所在:

    run方法

    从上面可以看到关键就是objc_msgSendSuper方法了,我们进入源码里面看看这个方法:

    objc_msgSendSuper

    可以看到这个super 是从父类中查找方法,不代表就是返回父类的方法,因此class方法又是因为在NSObject对象中,所以最终找到的都是在NSObject中,这个时候就查看class源码就可以知道结果了:

    class

    可以看到class 的方法返回的就是self,也就是方法调用接受者,就是上面的receiver字段,那就是self,所以上面的打印结果不言而喻了

    面试题2:

    代码图:

    Person 打印方法 调试方法

    为什么obj 能找到print方法呢?现在假设如果我们是通过person对象去调用又是怎样呢?

    方法调用图

    可以看到如果是person对象调用print 方法的话,是通过person->对象第一个变量地址,然后再调用print方法,同样obj也是一样,obj->对象第一个变量地址,然后找到person 类对象,进而调用print方法,所以[obj print] 也是能正确调用到方法的,至于为什么打印结果是123呢?

    因为局部变量是分配在栈空间的,而且内存是有大到小分布的,所以viewDidLoad方法里面的内存排布是:

    内存分布图

    可以知道print 就是打印出self->name 的值,换句话说,就是找到self这个对象,然后跳过对象的isa(8个字节,找到接下来的8个字节的变量值),下面Person_IMPL代码可以说明这点,上图中的cls也就相当于isa,那么obj通过cls找到了person对象,然后再找后面8个变量的字节,因为内存又是连续的,所以后面8个字节自然而言就是test变量,所以打印出来是123

           struct Person_IMPL

        {

               Class isa;

              NSString *_name;

         };

    但是如果去掉test这个变量的话,是不是会打印null呢,因为没有了变量?

    测试代码

    结果发现打印出来是ViewController,为什么呢?原来我们漏掉了两个默认参数,[super ViewDidLoad] 调用的时候,会默认传入两个参数,一个是结构体(前面abc那一串注释的),另外一个是_cmd,所以真正的内存布局应该是:

    内存布局

    这个时候的self 代表的就是消息的接受者,也就是ViewController,所以通过cls找到了对象之后,往下移动8个字节,就是self,所以打印结果是ViewController,但是其实上面一个图还是有点小问题的,就是第四个存放的应该是ViewController Class,虽然我们编译成C++代码的时候,可以看到调用super的时候调用的是objc_msgSendSuper 方法,但是这个源码只能提供参考,实际上在运行的时候调用的是objc_msgSendSuper2这个方法:

    运行汇编图:

    汇编图

    源码汇编图:

    汇编源码图

    可以看到这里会调用class-superclass,因此可以说明这里调用的应该是objc_msgSendSuper2,并且传递进去的结构体是{self,ViewController.Class},另外也可以通过lldb断点调试一下:

    lldb证明图

    objc_msgSendSuper2的确是真正调用,并且正在的内存图应该是:

    lldb调试1 lldb调试2 真正内存图

    补充

    super调用,底层会转换为objc_msgSendSuper2函数的调用,接受2个参数

    objc_super2

    相关文章

      网友评论

          本文标题:Super关键字

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