Super关键字

面试题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关键字)