• <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)編和動(dòng)態(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;
            }


            上面程序運(yùn)行會(huì)報(bào)錯(cuò)嗎?
            ——————————————————————

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

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

            要解此題首先要明確兩個(gè)問題。

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

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

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

            匯編如下:

            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()的時(shí)候只是把p的值移動(dòng)到了一個(gè)暫存器里面,但是并沒有用到這個(gè)值,后面就直接調(diào)用AA::result函數(shù)了,0x41105A正是該函數(shù)的入口地址。

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

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


            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)編"和"動(dòng)態(tài)聯(lián)編"這兩個(gè)術(shù)語的出處在哪?  回復(fù)  更多評論   

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

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

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

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

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

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

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

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

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

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


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

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

            @OwnWaterloo
            @OwnWaterloo
            @溪流

            大學(xué)里教過,很多書里面也有講.
            但是,
            不好意思,文章是轉(zhuǎn)的.,可能記錯(cuò)了,不記得是哪本書里講過.
              回復(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è)計(jì)基礎(chǔ)》,大一時(shí)候一門課用的  回復(fù)  更多評論   

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

            導(dǎo)航

            統(tǒng)計(jì)

            常用鏈接

            留言簿(4)

            隨筆分類(378)

            隨筆檔案(329)

            鏈接

            最新隨筆

            搜索

            最新評論

            国产一区二区三区久久精品| 久久九九全国免费| 少妇精品久久久一区二区三区| 国产成人精品综合久久久| 99久久精品日本一区二区免费| 日韩精品国产自在久久现线拍| 亚洲国产婷婷香蕉久久久久久 | 久久久久久亚洲Av无码精品专口 | 无码人妻久久一区二区三区| 99久久伊人精品综合观看| 国产69精品久久久久久人妻精品| 久久香蕉一级毛片| 伊人久久大香线蕉亚洲| 亚洲国产成人久久精品影视| 欧美一区二区三区久久综合 | 久久久婷婷五月亚洲97号色| 中文字幕精品无码久久久久久3D日动漫| 亚洲精品无码久久久久去q| 亚洲一区中文字幕久久| 2021国内精品久久久久久影院| 日本久久久久久中文字幕| 日本强好片久久久久久AAA| 亚洲精品tv久久久久| 国产精品成人久久久久久久| 麻豆精品久久精品色综合| 亚洲精品乱码久久久久久| 亚洲欧美成人久久综合中文网 | 久久成人小视频| 一级做a爰片久久毛片毛片| 久久久久久A亚洲欧洲AV冫| 久久综合色之久久综合| 久久福利片| 久久久精品久久久久久| 久久影院午夜理论片无码 | 精品伊人久久大线蕉色首页| 热久久最新网站获取| 狠狠综合久久综合88亚洲 | 99久久精品费精品国产一区二区| 久久亚洲精精品中文字幕| www性久久久com| 91精品国产91久久久久久蜜臀|