• <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 萬連文 閱讀(1277) 評論(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)

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

            <2007年5月>
            293012345
            6789101112
            13141516171819
            20212223242526
            272829303112
            3456789

            常用鏈接

            留言簿(66)

            隨筆分類

            隨筆檔案

            相冊

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            久久亚洲国产最新网站| 久久亚洲国产精品一区二区| 久久香蕉国产线看观看精品yw| 青春久久| 久久亚洲精品无码观看不卡| 精品久久久久久无码免费| 国产伊人久久| 久久精品亚洲男人的天堂| 久久人人爽人爽人人爽av| 久久亚洲AV无码西西人体| 香蕉99久久国产综合精品宅男自 | 久久国产欧美日韩精品| 亚洲国产成人精品91久久久 | 久久久久久a亚洲欧洲aⅴ| 国产精品久久久久无码av| 色综合久久中文综合网| 久久国产成人| 久久久久se色偷偷亚洲精品av| 欧洲精品久久久av无码电影| 国产成人精品白浆久久69| 国产综合精品久久亚洲| 久久亚洲精品成人无码网站 | 无码任你躁久久久久久老妇| 久久99久国产麻精品66| 久久91精品国产91久久小草| 观看 国产综合久久久久鬼色 欧美 亚洲 一区二区 | 久久久久久精品成人免费图片| 国产精品久久久久乳精品爆| 久久久久九九精品影院| 久久精品国产久精国产一老狼| 久久久久久国产精品免费无码 | 亚洲国产另类久久久精品黑人 | 久久婷婷色综合一区二区| 国产精品久久久久天天影视| 伊人情人综合成人久久网小说| 国产精品一区二区久久国产| 青草久久久国产线免观| 久久夜色tv网站| 久久婷婷国产综合精品| 欧美久久久久久精选9999| 狠狠色婷婷综合天天久久丁香|