• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            天下

            記錄修行的印記

            [轉(zhuǎn)]深入解析c++靜態(tài)聯(lián)編和類的成員函數(shù)調(diào)用

            只要學了C++的人,肯定知道靜態(tài)聯(lián)編和動態(tài)聯(lián)編,如果你不知道,ok那你學習 之路還長。簡單的靜態(tài)聯(lián)編的東西就不說了。先看下面程序。

            #include <iostream>
            using namespace std;
            class AA{
             
            public:
                 
            void result()
                   {
                        std::cout 
            << "Surprise?" << std::endl;
                   };
             };
            int main()
            {
                AA 
            *= NULL; //注意這里是NULL
                p
            ->result();
                ((AA
            *)0)->result();
                system(
            "Pause");
                
            return 0;
            }


            上面程序運行會報錯嗎?
            ——————————————————————

            如果你說運行一切正常并知道原因,ok。那就別往下看了,時間就是金錢。

            確實,這個運行正常并輸出  Surprise? 不信?你copy過去運行下試試。為啥啊。明明指針p的值是NULL,而你使用NULL指針去調(diào)用成員函數(shù),明明會報內(nèi)存錯誤的瑟。書上不是說了不能使用 NULL指針嗎?嘿嘿,沒錯,確實不能使用NULL指針,但是這里,程序根本就沒有用指針p的值,而是僅僅用到了它的類型做靜態(tài)束定而已。

            要解此題首先要明確兩個問題。

            1、靜態(tài)聯(lián)編的原理;2、成員函數(shù)的代碼在運行期只有一份拷貝。

            靜態(tài)聯(lián)編簡單的說就是在編譯期就已經(jīng)確定了要調(diào)用哪個函數(shù)了,這里的result()就是。同時要知道,類的成員函數(shù)在運行期只有一份拷貝在內(nèi)存,不管類的實例有多少個,成員函數(shù)始終只有一份代碼在內(nèi)存,因此只要知道類的指針的類型之后,就可以定位到函數(shù)的入口地址,根本不關(guān)心該指針指向的是一個什么東西。成員函數(shù)和成員變量不一樣,非靜態(tài)成員變量是跟隨類的實例走的。

            ok,明白上面兩個問題之后,這個事情就好解決了。直接上匯編吧。

            匯編如下:

            AA *p = NULL;
            00411ACE  mov         dword ptr [p],0
            p->result();
            00411AD5  mov         ecx,dword ptr [p]
            00411AD8  call          AA::result (41105Ah)

            清楚了吧。在執(zhí)行p->result()的時候只是把p的值移動到了一個暫存器里面,但是并沒有用到這個值,后面就直接調(diào)用AA::result函數(shù)了,0x41105A正是該函數(shù)的入口地址。

            ok,好了。不僅可以向以上說的去訪問成員函數(shù),甚至再過分一點((A*)0)->result();這樣都可以。你再火一點把那個0換成任意一個地址都可以正確調(diào)用到那個函數(shù),因為編譯器在靜態(tài)束定的時候只關(guān)心那個指針的類型。當然了,不可這樣去訪問類的成員變量,因為成員變量是在對象的內(nèi)存布局里面的。

            值得說一點的是,如果你在result函數(shù)里面有涉及到類的成員變量的訪問,那么這顯然就會出錯了,因為成員變量需要通過傳進來的this指針(其實就可以理解成時p指針)去訪問對象的內(nèi)存的。然而此時p還沒有指向一個有效的空間。故而出錯。


            posted on 2010-10-26 12:02 天下 閱讀(998) 評論(8)  編輯 收藏 引用 所屬分類: C/C++

            評論

            # re: 深入解析c++靜態(tài)聯(lián)編和類的成員函數(shù)調(diào)用 2010-10-26 18:26 OwnWaterloo

            請教一下, "靜態(tài)聯(lián)編"和"動態(tài)聯(lián)編"這兩個術(shù)語的出處在哪?  回復(fù)  更多評論   

            # re: 深入解析c++靜態(tài)聯(lián)編和類的成員函數(shù)調(diào)用 2010-10-26 23:46 ...

            這跟C++有什么關(guān)系? 內(nèi)存儲存機制都是這樣的。 你間接調(diào)用一個API 看看, 看看真正的API入口點是什么。

            甚至再過分一點((A*)0)->result();這樣都可以。你再火一點把那個0換成任意一個地址都可以正確調(diào)用到那個函數(shù)

            你在函數(shù)上面加個變量看看。  回復(fù)  更多評論   

            # re: 深入解析c++靜態(tài)聯(lián)編和類的成員函數(shù)調(diào)用 2010-10-27 09:14 天下

            @OwnWaterloo
            《深度探索C++對象模型》有詳細的解釋.
              回復(fù)  更多評論   

            # re: 深入解析c++靜態(tài)聯(lián)編和類的成員函數(shù)調(diào)用 2010-10-27 10:08 溪流

            @天下
            沒看出來質(zhì)疑的語氣嗎?
            不過很多大學教科書確實也都這么叫。。。  回復(fù)  更多評論   

            # re: 深入解析c++靜態(tài)聯(lián)編和類的成員函數(shù)調(diào)用 2010-10-27 11:01 OwnWaterloo

            @天下
            能詳細說說是哪一章節(jié), 以及哪一版本嗎?
            中文版至少有3個版本, 侯捷的簡體與繁體、 還有另一個人翻譯的簡體。
            再次感謝~


            @溪流
            我最開始是聽一個學弟在論壇上這么說, 以為是他自己造的一個詞。
            后來偶爾也有文章出現(xiàn)這樣的詞。
            今個兒終于逮到機會問問。
            是哪本大學教科書啊?  回復(fù)  更多評論   

            # re: 深入解析c++靜態(tài)聯(lián)編和類的成員函數(shù)調(diào)用 2010-10-27 16:27 天下

            @OwnWaterloo
            @OwnWaterloo
            @溪流

            大學里教過,很多書里面也有講.
            但是,
            不好意思,文章是轉(zhuǎn)的.,可能記錯了,不記得是哪本書里講過.
              回復(fù)  更多評論   

            # re: [轉(zhuǎn)]深入解析c++靜態(tài)聯(lián)編和類的成員函數(shù)調(diào)用[未登錄] 2010-10-27 23:25 OwnWaterloo

            @天下
            I see. 謝謝~  回復(fù)  更多評論   

            # re: [轉(zhuǎn)]深入解析c++靜態(tài)聯(lián)編和類的成員函數(shù)調(diào)用 2010-10-28 02:16 溪流

            @OwnWaterloo
            我手上有一本科學出版社《Visual C++ 程序設(shè)計基礎(chǔ)》,大一時候一門課用的  回復(fù)  更多評論   

            <2025年8月>
            272829303112
            3456789
            10111213141516
            17181920212223
            24252627282930
            31123456

            導(dǎo)航

            統(tǒng)計

            常用鏈接

            留言簿(4)

            隨筆分類(378)

            隨筆檔案(329)

            鏈接

            最新隨筆

            搜索

            最新評論

            97久久精品人人做人人爽| 狠狠色婷婷久久一区二区三区| 91久久国产视频| 久久综合色区| 久久丫精品国产亚洲av| 欧美激情精品久久久久| 欧美久久天天综合香蕉伊| 精品无码久久久久久尤物| 久久国产福利免费| 99久久精品午夜一区二区| 久久亚洲精品无码观看不卡| 久久男人Av资源网站无码软件 | 久久天堂电影网| 久久成人国产精品免费软件| 91精品国产色综久久| 久久人人爽人人爽人人AV| 久久无码人妻精品一区二区三区 | 无码人妻少妇久久中文字幕蜜桃 | 国产成人香蕉久久久久| 久久久久久国产精品免费无码| 久久综合伊人77777麻豆| 青青草原1769久久免费播放| 蜜臀av性久久久久蜜臀aⅴ | 亚洲国产成人久久一区久久| 色综合久久中文综合网| 狠狠色丁香久久婷婷综合五月| 久久亚洲熟女cc98cm| 久久亚洲精品无码aⅴ大香 | 大蕉久久伊人中文字幕| 久久精品国产精品青草| 狠狠色丁香久久婷婷综| 国产91色综合久久免费| 国产一区二区三区久久精品| 国产精品禁18久久久夂久| 久久棈精品久久久久久噜噜| 日韩人妻无码精品久久免费一| 亚洲中文久久精品无码ww16| 无码国内精品久久人妻蜜桃| 国产婷婷成人久久Av免费高清 | 国内精品久久久久影院一蜜桃 | 亚洲日韩欧美一区久久久久我|