Runtime

神奇的 self.name

sunnyxx 的一個 runtime 線下分享里面, 有一道試題:

@interface Sark : NSObject
@property(nonatomic, copy) NSString *name;
@end
@implementation Sark
-(void)speak {
    NSLog(@"hello, %@", self.name);
}
@end
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        NSString *name = @"Sunnyxx";
        id cls = [Sark class];
        void *obj = &cls;
        [(__bridge id)obj speak];
    }
    return 0;
}

這段代碼運(yùn)行結(jié)果是 hello, Sunnyxx
是不是感覺白學(xué)了 iOS 開發(fā)這么久?
別擔(dān)心, 我們一步一步來分析代碼!
首先NSString *name = @"Sunnyxx"; 這個簡直莫名其妙, 創(chuàng)建一個對象, 又不用, 結(jié)果還能打印出來, 簡直看不懂.開始懵逼了
嗯, 第二句, 這個簡單, 獲取 class.
NSObject 中這個有源代碼, 注意和實例方法-class 不同的! 不過這個看起來也簡單易懂, 返回自己(類對象)

//https://opensource.apple.com/source/objc4/objc4-709/runtime/NSObject.mm.auto.html
+ (Class)class {
    return self;
}

接下來void *obj = &cls; 獲取類對象的地址? 有啥用?? 越看越懵逼.
[(__bridge id)obj speak]; 這又是啥, 為啥可以調(diào)用 speak 方法?還可以打印莫名其妙的變量?
答案要從 OBJC 的對象本身說起.
OBJC 中的對象(id)都是這個東西
typedef struct objc_object *id;
objc_object 是這樣定義的

https://opensource.apple.com/source/objc4/objc4-709/runtime/objc-private.h.auto.html
struct objc_object {
private:
    isa_t isa;
    ...
}

可以看到, isa 指針是 objc_object 的第一個屬性,
而對象 alloc 之后, 會將 isa 指針指向類對象, 可以看著里
比如id instance = [Sark alloc] 這段代碼
我們假設(shè), Sark 類對象在內(nèi)存中的地址是0x0030, alloc 出的對象地址是0x0020, instance 本身的地址是0x0010, 而 isa 指針是 objc_object 的第一個屬性, 那么 isa 指針的地址也是0x0010, 整理一下可得
instance(0x0010)-指向>isa指針/創(chuàng)建出來的 Sark 實例對象(0x0020)-指向>Sark 類對象(0x0030);
對比一下之前的代碼void *obj = &cls;, 假設(shè), obj 本身的地址是0x0050
然后
obj(0x0050)-指向>Sark 類對象(0x0030);
這個 obj 是不是和上面那個實例對象有點像, 都是指向 Sark 類對象的指針.
[xxx speak]最終是到 Sark 類對象中查找 speak 方法的實現(xiàn).
而obj 和對象都是一樣的, 指向 Sark 類對象, 所以調(diào)用 speak 方法是沒有問題的.
那為啥可以打印出 "hello, Sunnyxx"
"Sunnyxx" 是 只是一個局部變量, 跟 Sark 有什么關(guān)系?
一個對象創(chuàng)建好之后, 基本上結(jié)構(gòu)是這樣的


可以看到, name 這個屬性, 就是在 isa 指針后的.
可以寫個代碼驗證一下

Sark* instance = [Sark new];
instance.name = @"world";
void **namepp = ((__bridge void*)instance + 8);
NSString *namep = (__bridge NSString *)(*namepp);
NSLog(@"instance name is %@", namep);

這段代碼可以正常打印出instance name is world
可以看出來, name 和 instance 剛好偏移了8個字節(jié), 也就是64位系統(tǒng)下一個指針的大小. 這也說明了一個實例對象的屬性存儲方式, 對象在訪問屬性 _name 的時候, 其實就是去找離自己偏移量為 8 的變量.
所以呢?
我們回到這里

NSString *name = @"Sunnyxx";
id cls = [Sark class];
void *obj = &cls;
[(__bridge id)obj speak];

這段最古怪的代碼: 局部變量 name的地址和 cls自身的地址之間也是相差8個字節(jié), 即 name 的地址 = &cls + 8( 為什么是+ 不是 - , 因為這里是局部變量, 在棧中存儲, 越早定義的變量地址越高)
是不是和剛才有點像?有點懂了?
再來理一下.
當(dāng)在 speak 方法中去調(diào)用 self.name 的時候, 去找離 self 為8 字節(jié)的變量, 和 self 距離為8 的變量就是這個局部變量 name, 于是就把 Sunnyxx 打印出來了

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,117評論 6 537
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,860評論 3 423
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,128評論 0 381
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,291評論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 72,025評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,421評論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,477評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,642評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,177評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 40,970評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,157評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,717評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,410評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,821評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,053評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,896評論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 48,157評論 2 375

推薦閱讀更多精彩內(nèi)容