• <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)用

            只要學(xué)了C++的人,肯定知道靜態(tài)聯(lián)編和動態(tài)聯(lián)編,如果你不知道,ok那你學(xué)習(xí) 之路還長。簡單的靜態(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)心那個指針的類型。當(dāng)然了,不可這樣去訪問類的成員變量,因為成員變量是在對象的內(nèi)存布局里面的。

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


            posted on 2010-10-26 12:02 天下 閱讀(996) 評論(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ì)疑的語氣嗎?
            不過很多大學(xué)教科書確實也都這么叫。。。  回復(fù)  更多評論   

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

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


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

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

            @OwnWaterloo
            @OwnWaterloo
            @溪流

            大學(xué)里教過,很多書里面也有講.
            但是,
            不好意思,文章是轉(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
            我手上有一本科學(xué)出版社《Visual C++ 程序設(shè)計基礎(chǔ)》,大一時候一門課用的  回復(fù)  更多評論   

            <2012年3月>
            26272829123
            45678910
            11121314151617
            18192021222324
            25262728293031
            1234567

            導(dǎo)航

            統(tǒng)計

            常用鏈接

            留言簿(4)

            隨筆分類(378)

            隨筆檔案(329)

            鏈接

            最新隨筆

            搜索

            最新評論

            日本久久久久久久久久| 色综合合久久天天综合绕视看| 久久人妻少妇嫩草AV蜜桃| 亚洲&#228;v永久无码精品天堂久久| 午夜精品久久久久久影视777| 伊人久久无码中文字幕| 久久99精品国产麻豆宅宅| 久久国产高清一区二区三区| 久久九九兔免费精品6| Xx性欧美肥妇精品久久久久久| 日批日出水久久亚洲精品tv| 精品少妇人妻av无码久久| 欧洲性大片xxxxx久久久| 国产精品99久久免费观看| 一级女性全黄久久生活片免费| 丁香狠狠色婷婷久久综合| 亚洲婷婷国产精品电影人久久| 国产欧美久久一区二区| 久久99精品国产麻豆宅宅| 久久九色综合九色99伊人| 久久精品中文字幕久久| 久久影院综合精品| 伊人色综合久久天天人守人婷 | 久久精品水蜜桃av综合天堂| 久久久久亚洲精品男人的天堂| 久久狠狠色狠狠色综合| 久久亚洲中文字幕精品有坂深雪 | 久久国产亚洲精品| 久久精品成人免费国产片小草| a高清免费毛片久久| 久久久久亚洲精品无码蜜桃| 新狼窝色AV性久久久久久| 狠狠色婷婷久久综合频道日韩| 久久亚洲国产精品123区| 久久精品国产亚洲Aⅴ蜜臀色欲| 四虎国产永久免费久久| 色综合色天天久久婷婷基地| 久久精品成人免费看| 久久国产乱子伦精品免费午夜| 国产精品成人无码久久久久久| 久久精品国产亚洲网站|