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

            隨筆分類

            隨筆檔案

            相冊

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            久久精品国产亚洲Aⅴ香蕉| 国产精品欧美久久久久天天影视 | 久久精品无码一区二区三区| 婷婷久久五月天| 久久精品国产亚洲7777| 情人伊人久久综合亚洲| 2021久久国自产拍精品| 久久精品国产99久久无毒不卡| 国产精品亚洲综合久久| 精品久久久久久久国产潘金莲| 日日狠狠久久偷偷色综合免费| AAA级久久久精品无码区| 女人香蕉久久**毛片精品| 久久精品国产免费| 成人精品一区二区久久| 国产一区二区精品久久凹凸 | 亚洲综合婷婷久久| 97精品国产97久久久久久免费| 久久精品一区二区三区不卡| 久久综合狠狠色综合伊人| 久久精品国产精品国产精品污| 色综合久久天天综合| 国产免费久久久久久无码| 久久夜色精品国产亚洲| 久久93精品国产91久久综合| 久久久久香蕉视频| 久久中文字幕人妻丝袜| 久久综合香蕉国产蜜臀AV| 久久久久久免费一区二区三区| 国产一区二区三精品久久久无广告| 久久久91人妻无码精品蜜桃HD| 久久久久久久91精品免费观看| 一本久久a久久精品vr综合| 国内精品伊人久久久久av一坑| 国产日韩久久久精品影院首页| 理论片午午伦夜理片久久| 模特私拍国产精品久久| 国产午夜精品久久久久免费视| 久久精品国产精品亚洲艾草网美妙 | 久久国产精品无| 9久久9久久精品|