面試題1:
上代碼圖:
Student.m
從打印結(jié)果里面看到super class 返回的是Student猜旬,而[super superclass] 返回Person脚曾,是不是感覺有點奇怪?現(xiàn)在我們就通過重寫run方法悉盆,調(diào)用[super run] 查看super的本質(zhì)
xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc Student.m 然后進(jìn)入Student.cpp 文件里面找到run方法所在:
從上面可以看到關(guān)鍵就是objc_msgSendSuper方法了鄙麦,我們進(jìn)入源碼里面看看這個方法:
可以看到這個super 是從父類中查找方法典唇,不代表就是返回父類的方法镊折,因此class方法又是因為在NSObject對象中,所以最終找到的都是在NSObject中介衔,這個時候就查看class源碼就可以知道結(jié)果了:
可以看到class 的方法返回的就是self恨胚,也就是方法調(diào)用接受者,就是上面的receiver字段炎咖,那就是self赃泡,所以上面的打印結(jié)果不言而喻了
面試題2:
代碼圖:
為什么obj 能找到print方法呢?現(xiàn)在假設(shè)如果我們是通過person對象去調(diào)用又是怎樣呢乘盼?
可以看到如果是person對象調(diào)用print 方法的話升熊,是通過person->對象第一個變量地址,然后再調(diào)用print方法蹦肴,同樣obj也是一樣僚碎,obj->對象第一個變量地址,然后找到person 類對象阴幌,進(jìn)而調(diào)用print方法勺阐,所以[obj print] 也是能正確調(diào)用到方法的,至于為什么打印結(jié)果是123呢矛双?
因為局部變量是分配在椩ǔ椋空間的,而且內(nèi)存是有大到小分布的议忽,所以viewDidLoad方法里面的內(nèi)存排布是:
可以知道print 就是打印出self->name 的值懒闷,換句話說,就是找到self這個對象栈幸,然后跳過對象的isa(8個字節(jié)愤估,找到接下來的8個字節(jié)的變量值),下面Person_IMPL代碼可以說明這點,上圖中的cls也就相當(dāng)于isa速址,那么obj通過cls找到了person對象玩焰,然后再找后面8個變量的字節(jié),因為內(nèi)存又是連續(xù)的芍锚,所以后面8個字節(jié)自然而言就是test變量昔园,所以打印出來是123
? ? ? ?struct Person_IMPL
? ? {
? ? ? ? ? ?Class isa;
? ? ? ? ? NSString *_name;
? ? ?};
但是如果去掉test這個變量的話,是不是會打印null呢并炮,因為沒有了變量默刚?
結(jié)果發(fā)現(xiàn)打印出來是ViewController,為什么呢逃魄?原來我們漏掉了兩個默認(rèn)參數(shù)荤西,[super ViewDidLoad] 調(diào)用的時候,會默認(rèn)傳入兩個參數(shù),一個是結(jié)構(gòu)體(前面abc那一串注釋的)皂冰,另外一個是_cmd,所以真正的內(nèi)存布局應(yīng)該是:
這個時候的self 代表的就是消息的接受者店展,也就是ViewController养篓,所以通過cls找到了對象之后秃流,往下移動8個字節(jié),就是self柳弄,所以打印結(jié)果是ViewController舶胀,但是其實上面一個圖還是有點小問題的,就是第四個存放的應(yīng)該是ViewController Class碧注,雖然我們編譯成C++代碼的時候嚣伐,可以看到調(diào)用super的時候調(diào)用的是objc_msgSendSuper 方法,但是這個源碼只能提供參考萍丐,實際上在運行的時候調(diào)用的是objc_msgSendSuper2這個方法:
運行匯編圖:
源碼匯編圖:
可以看到這里會調(diào)用class-superclass,因此可以說明這里調(diào)用的應(yīng)該是objc_msgSendSuper2轩端,并且傳遞進(jìn)去的結(jié)構(gòu)體是{self,ViewController.Class},另外也可以通過lldb斷點調(diào)試一下:
objc_msgSendSuper2的確是真正調(diào)用,并且正在的內(nèi)存圖應(yīng)該是:
補充
super調(diào)用逝变,底層會轉(zhuǎn)換為objc_msgSendSuper2函數(shù)的調(diào)用基茵,接受2個參數(shù)