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

            鏈接

            最新隨筆

            搜索

            最新評論

            久久久噜噜噜www成人网| 日韩久久无码免费毛片软件| 日本亚洲色大成网站WWW久久| 尹人香蕉久久99天天拍| 狠狠色丁香婷综合久久| 久久99这里只有精品国产| 久久夜色精品国产亚洲| 久久亚洲精精品中文字幕| 久久这里只有精品首页| 久久影视国产亚洲| 久久精品国产亚洲精品2020| 久久国产综合精品五月天| 久久婷婷色综合一区二区| 精品国产乱码久久久久久郑州公司| 99久久精品影院老鸭窝| 精品久久久无码中文字幕| 伊人久久大香线蕉综合Av| 国产视频久久| 欧美激情精品久久久久| 亚洲中文字幕久久精品无码APP| 国产精品青草久久久久婷婷| 午夜精品久久久久久影视riav| 99久久国产免费福利| 无码超乳爆乳中文字幕久久| 精品久久久久久久国产潘金莲 | 97久久国产亚洲精品超碰热| 精品国产91久久久久久久| 伊人热人久久中文字幕| 欧美国产成人久久精品| 中文精品99久久国产| 国产精品一久久香蕉国产线看| 精品久久久久久国产91| 色老头网站久久网| 精品久久久久久无码国产| 久久伊人精品一区二区三区| 久久99精品免费一区二区| 久久99国产精品久久99| 久久被窝电影亚洲爽爽爽| 亚洲欧洲精品成人久久奇米网| 国产免费福利体检区久久| 久久精品国产精品亚洲毛片|