• <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>
            萬星星@豌豆莢 歡迎加入我們
            一個吃軟飯的男人!!!!!我只想寫程序####
            微博:http://weibo.com/wanlianwen
            posts - 172,  comments - 1253,  trackbacks - 0

            經常存在這樣一種情況,寫代碼的時候按照一種固有思維去寫,可能時間久了都沒有考慮過為什么這么寫,不這么寫會引起什么錯誤。我時常有這樣的困惑,了解太少,有時候遇到蹊蹺的事情也沒時間探究。
            今天在跟蹤mfc庫的時候,遇到一件事情使我非常驚訝,徹底打破了我對空對象指針的看法。在我的概念里面,空的對象指針是不能使用的,不能調用方法。不知道大家有沒有這樣的想法,事實證明是不對的。下面看看我跟蹤的代碼:

            CWnd * ?PASCAL?CWnd::FromHandle(HWND?hWnd)
            {
            ????CHandleMap
            * ?pMap? = ?afxMapHWND(TRUE);? // create?map?if?not?exist
            ????ASSERT(pMap? != ?NULL);
            ????CWnd
            * ?pWnd? = ?(CWnd * )pMap -> FromHandle(hWnd);

            #ifndef?_AFX_NO_OCC_SUPPORT
            ????pWnd
            -> AttachControlSite(pMap);
            #endif

            ????ASSERT(pWnd?
            == ?NULL? || ?pWnd -> m_hWnd? == ?hWnd);
            ????
            return ?pWnd;
            }
            上面代碼中,pWnd指針是空的,然而可以調用方法。我百思不得其解,無奈只有寫一個例子看看。
            #include?<iostream>
            using?namespace?std;

            class?CObj
            {
            public:
            ????
            void?Test();
            public:
            ????
            static?int?m_iInt;
            }
            ;

            int?CObj::m_iInt?=?100;

            void?CObj::Test()
            {
            ????cout?
            <<?m_iInt?<<?endl;
            }


            int?main(int?argc,?char*?argv[])
            {
            ????CObj
            *????????pObj????=?(CObj*)1;
            ????pObj
            ->Test();
            ????((CObj
            *)0)->Test();
            ????((CObj
            *)1)->Test();
            ????CObj????????obj;
            ????obj.Test();
            ????
            return?0;
            }
            一個類,包含一個靜態變量和一個實例方法。可以通過空指針,非法指針訪問靜態變量和實例方法,前提是方法里面不可以訪問成員變量,否則在運行期出現非法訪問錯誤,因為對象在內存中不存在,而方法是存在的。我開始懷疑靜態方法的威力了,它只不過提供了編譯期成員的安全訪問罷了。

            下面看看匯編,因為只有通過匯編才能了解更深的問題:
            25:???????CObj*???????pObj????=?(CObj*)1;
            00401848???mov?????????dword?ptr?[ebp-4],1
            26:???????pObj->Test();
            0040184F???mov?????????ecx,dword?ptr?[ebp
            -4]
            00401852???call????????@ILT+565(CObj::Test)?(0040123a)
            27:???????((CObj*)0)->Test();
            00401857???xor?????????ecx,ecx
            00401859???call????????@ILT+565(CObj::Test)?(0040123a)
            28:???????((CObj*)1)->Test();
            0040185E???mov?????????ecx,
            1
            00401863???call????????@ILT+565(CObj::Test)?(0040123a)
            29:???????CObj????????obj;
            30:???????obj.Test();
            00401868???lea?????????ecx,[ebp-8]
            0040186B???call????????@ILT
            +565(CObj::Test)?(0040123a)
            俺不懂匯編,臨時查了一下,獻丑了。
            ecx里面存放的是對象的this指針,@ILT是函數的入口點,調用時將ecx傳遞到函數內部,其實對于編譯器函數是全局的,函數調用的時候都需要傳遞this指針作為第一個參數傳遞過去。所以可以上面的代碼在內存訪問方面都是合法的,一旦訪問this的非靜態成員變量,勢必導致非法內存訪問。

            順便侃侃編譯器優化,我不懂到底那個操作效率高,只談談區別。
            ((CObj*)0)->Test();((CObj*)1)->Test();對應的匯編:
            00401857???xor?????????ecx,ecx
            00401859???call????????@ILT+565(CObj::Test)?(0040123a)

            0040185E???mov?????????ecx,
            1
            00401863???call????????@ILT+565(CObj::Test)?(0040123a)
            靠,這都有區別,0指針的時候異或一下剛好是0。無奈啊,編譯器確實好智能。
            posted on 2006-09-20 11:49 萬連文 閱讀(1279) 評論(3)  編輯 收藏 引用 所屬分類: 亂七八糟

            FeedBack:
            # re: 由一個空對象指針引起的思考
            2006-09-20 12:23 | ^_^
            靜態方法當然可以調用啦……
            匯編生成0在我的記憶中編譯器一直是這樣做的。  回復  更多評論
              
            # re: 由一個空對象指針引起的思考
            2006-09-20 12:45 | LOGOS

            呵呵。如果從匯編,甚至機器碼的角度來看待程序,那么,很多語言特性將不復存在。因為這些特性都是在編譯期得到保證的,比如:public, const等。
            這也就是,很多時候憑著一個指針就可以在C/C++中走天下了。
            不過對于接口類(全是純虛函數),透過指針的操作要三思而行。  回復  更多評論
              
            # re: 由一個空對象指針引起的思考
            2006-09-20 14:00 | eXile
            空對象指針主要用在一些 hacker 方面,比如取得類成員的偏移量:
            #define offsetof(ClassType, Member) (size_t) &(((ClassType*)0)->Member)

            不過我覺得還是少用為妙
              回復  更多評論
              
            簡歷下載
            聯系我

            <2006年9月>
            272829303112
            3456789
            10111213141516
            17181920212223
            24252627282930
            1234567

            常用鏈接

            留言簿(66)

            隨筆分類

            隨筆檔案

            相冊

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            久久精品99久久香蕉国产色戒| 亚洲成色999久久网站| 久久99久久成人免费播放| 国产高潮国产高潮久久久91 | 久久er99热精品一区二区| 久久久久久午夜成人影院 | 久久精品国产亚洲网站| 国产日韩久久免费影院| 欧美激情一区二区久久久| 国产成人综合久久综合| 久久精品免费全国观看国产| 无码人妻久久久一区二区三区| 91精品无码久久久久久五月天| 久久人人爽人人爽人人片AV不 | 久久综合狠狠综合久久97色| 伊人久久大香线蕉综合影院首页 | 久久久久亚洲AV无码专区首JN| 狠狠久久亚洲欧美专区| 久久精品日日躁夜夜躁欧美| 国产精品成人99久久久久91gav| 久久久久亚洲AV片无码下载蜜桃 | 国内精品人妻无码久久久影院导航 | 久久这里只有精品首页| 精品国产青草久久久久福利| 亚洲精品美女久久久久99| 久久久久亚洲精品中文字幕| 精品国产一区二区三区久久| 久久99国产综合精品免费| 久久久久99这里有精品10| 欧美精品福利视频一区二区三区久久久精品 | 久久久久人妻一区二区三区| 久久久网中文字幕| 嫩草影院久久99| 久久久久久狠狠丁香| AV色综合久久天堂AV色综合在| 久久精品国产2020| 亚洲精品乱码久久久久久久久久久久 | 品成人欧美大片久久国产欧美 | 亚洲精品99久久久久中文字幕| 国产视频久久| 99热都是精品久久久久久|