面试题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
网友评论