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

            經(jīng)常存在這樣一種情況,寫代碼的時候按照一種固有思維去寫,可能時間久了都沒有考慮過為什么這么寫,不這么寫會引起什么錯誤。我時常有這樣的困惑,了解太少,有時候遇到蹊蹺的事情也沒時間探究。
            今天在跟蹤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;
            }
            一個類,包含一個靜態(tài)變量和一個實例方法。可以通過空指針,非法指針訪問靜態(tài)變量和實例方法,前提是方法里面不可以訪問成員變量,否則在運行期出現(xiàn)非法訪問錯誤,因為對象在內(nèi)存中不存在,而方法是存在的。我開始懷疑靜態(tài)方法的威力了,它只不過提供了編譯期成員的安全訪問罷了。

            下面看看匯編,因為只有通過匯編才能了解更深的問題:
            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是函數(shù)的入口點,調用時將ecx傳遞到函數(shù)內(nèi)部,其實對于編譯器函數(shù)是全局的,函數(shù)調用的時候都需要傳遞this指針作為第一個參數(shù)傳遞過去。所以可以上面的代碼在內(nèi)存訪問方面都是合法的,一旦訪問this的非靜態(tài)成員變量,勢必導致非法內(nèi)存訪問。

            順便侃侃編譯器優(yōu)化,我不懂到底那個操作效率高,只談談區(qū)別。
            ((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)
            靠,這都有區(qū)別,0指針的時候異或一下剛好是0。無奈啊,編譯器確實好智能。
            posted on 2006-09-20 11:49 萬連文 閱讀(1277) 評論(3)  編輯 收藏 引用 所屬分類: 亂七八糟

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

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

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

            <2007年9月>
            2627282930311
            2345678
            9101112131415
            16171819202122
            23242526272829
            30123456

            常用鏈接

            留言簿(66)

            隨筆分類

            隨筆檔案

            相冊

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            亚洲综合久久综合激情久久| 精品久久777| 久久精品国产99久久久| 久久电影网一区| 久久久久国产亚洲AV麻豆| 99精品久久精品一区二区| 亚洲国产精品久久| 色欲久久久天天天综合网精品| 久久精品一区二区| 区久久AAA片69亚洲 | 一本色道久久综合| 热久久国产精品| 久久久精品2019免费观看| 久久久久亚洲精品无码网址 | 一本大道久久香蕉成人网| 久久精品九九亚洲精品| 久久精品国产清自在天天线 | 国内精品九九久久久精品| 久久精品视频一| 国产精品永久久久久久久久久 | 国产精品热久久无码av| 精品乱码久久久久久久| 99久久国产宗和精品1上映| 亚洲午夜无码AV毛片久久| 久久97久久97精品免视看| 91精品国产综合久久香蕉 | 久久黄色视频| 婷婷综合久久中文字幕| 中文精品久久久久国产网址| 精品久久久无码人妻中文字幕豆芽| 无码人妻久久久一区二区三区| 一本综合久久国产二区| 青青热久久国产久精品 | 色综合久久久久无码专区| 亚洲综合熟女久久久30p| 久久精品国产一区二区| 亚洲国产精品成人久久蜜臀| 四虎久久影院| 亚洲AV日韩精品久久久久久久| 99精品久久久久久久婷婷| 99久久免费国产精品热|