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

            無我

            讓內(nèi)心永遠(yuǎn)燃燒著偉大的光明的精神之火!
            靈活的思考,嚴(yán)謹(jǐn)?shù)膶崿F(xiàn)
            豪邁的氣魄、頑強(qiáng)的意志和周全的思考

            關(guān)于C++虛函數(shù)的研究

                    以前早稍有接觸C++中虛函數(shù)多態(tài)在編譯器上的實現(xiàn)問題,金山面試時還著重問了這個問題。但總沒有利用程序來檢查實驗。這次終于對虛函數(shù)和他的vtable表有更深入的關(guān)注,通過繼承類大小的檢測和反匯編后的代碼能查出端倪,例如測試程序如下:

            #include <conio.h>
            #include <iostream>
            using namespace std;

            class A
            {
            private:
             char is[20];
            public:
             char ss[20];
             void f1() {cout<<"This A.f1"<<endl; }
             virtual void f2()  { cout<<"This A.f2"<<endl; }
             virtual void f3()  { cout<<"This A.f3"<<endl; }
            };
            class B:public A
            {
             void f1() {cout<<"This B.f1"<<endl; }
             void f2()  { cout<<"This B.f2"<<endl; }
             //void f3()  { cout<<"This B.f3"<<endl; }
            };

            void main()
            {
             cout<<sizeof(A)<<endl;
             cout<<sizeof(B)<<endl;
             A a,*p;
             B b;
             if(getch() == 'c')
              p = &b;
             else
              p = &a;
             p->f1();
             p->f2();
             p->f3();
            }

            發(fā)現(xiàn)sizeof(A)=sizeof(B)=44,除了數(shù)組共40個字節(jié)外,還有4個字節(jié)多于。最終調(diào)試發(fā)現(xiàn),只要父類有虛函數(shù),而又無論函數(shù)個數(shù)是多少,父類和所有繼承于他的子類都多4個字節(jié),做什么用的呢?存放一個指針,指向vtable表,反匯編顯示:

            32:       p->f1();
            00401626   mov         ecx,dword ptr [ebp-30h]
            00401629   call        @ILT+325(A::f1) (0040114a)
            33:       p->f2();
            0040162E   mov         edx,dword ptr [ebp-30h]
            00401631   mov         eax,dword ptr [edx]
            00401633   mov         esi,esp
            00401635   mov         ecx,dword ptr [ebp-30h]
            00401638   call        dword ptr [eax]
            0040163A   cmp         esi,esp
            0040163C   call        __chkesp (00420bc0)
            34:       p->f3();
            00401641   mov         ecx,dword ptr [ebp-30h]
            00401644   mov         edx,dword ptr [ecx]
            00401646   mov         esi,esp
            00401648   mov         ecx,dword ptr [ebp-30h]
            0040164B   call        dword ptr [edx+4]
            0040164E   cmp         esi,esp
            00401650   call        __chkesp (00420bc0)
            35:   }

            到00401638   call        dword ptr [eax]出F11進(jìn)入跳轉(zhuǎn)表:

            @ILT+560(?f2@B@@EAEXXZ):
            00401235   jmp         B::f2 (00401860)
            @ILT+565(?_Iput@?$num_put@DV?$ostreambuf_iterator@DU?$char_traits@D@std@@@std@@@std@@KA?AV?$ostreambuf_iterator@DU?$char_traits@D@std@@@2@V32@AAVios_base@2@DPADI@Z):
            0040123A   jmp         std::num_put<char,std::ostreambuf_iterator<char,std::char_traits<char> > >::_Iput (00404
            @ILT+570(_main):
            0040123F   jmp         main (004015b0)
            @ILT+575(??_Gfacet@locale@std@@UAEPAXI@Z):
            00401244   jmp         std::locale::facet::`scalar deleting destructor' (004035f0)
            @ILT+580(?_Allocate@std@@YAPADHPAD@Z):
            00401249   jmp         std::_Allocate (00406200)
            @ILT+585(??0?$num_put@DV?$ostreambuf_iterator@DU?$char_traits@D@std@@@std@@@std@@QAE@I@Z):
            0040124E   jmp         std::num_put<char,std::ostreambuf_iterator<char,std::char_traits<char> > >::num_put<char
            @ILT+590(?do_put@?$num_put@DV?$ostreambuf_iterator@DU?$char_traits@D@std@@@std@@@std@@MBE?AV?$ostreambuf_iterator@DU?$char_traits@D@std@@@2@V32@AAVios_base@2@D_N@Z):
            00401253   jmp         std::num_put<char,std::ostreambuf_iterator<char,std::char_traits<char> > >::do_put (0040
            @ILT+595(?put@?$basic_ostream@DU?$char_traits@D@std@@@std@@QAEAAV12@D@Z):
            00401258   jmp         std::basic_ostream<char,std::char_traits<char> >::put (00402580)
            @ILT+600(?max_size@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEIXZ):
            0040125D   jmp         std::basic_string<char,std::char_traits<char>,std::allocator<char> >::max_size (00405970
            @ILT+605(??0bad_cast@std@@QAE@PBD@Z):
            00401262   jmp         std::bad_cast::bad_cast (004031b0)
            @ILT+610(?size@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEIXZ):
            00401267   jmp         std::basic_string<char,std::char_traits<char>,std::allocator<char> >::size (00403a10)
            @ILT+615(?truename@?$numpunct@D@std@@QBE?AV?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@2@XZ):
            0040126C   jmp         std::numpunct<char>::truename (004054e0)
            @ILT+620(?allocate@?$allocator@D@std@@QAEPADIPBX@Z):
            00401271   jmp         std::allocator<char>::allocate (00405f60)
            @ILT+625(?to_int_type@?$char_traits@D@std@@SAHABD@Z):
            00401276   jmp         std::char_traits<char>::to_int_type (00402850)
            @ILT+630(?_Split@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@AAEXXZ):
            0040127B   jmp         std::basic_string<char,std::char_traits<char>,std::allocator<char> >::_Split (00405b90)
            @ILT+635(?sputn@?$basic_streambuf@DU?$char_traits@D@std@@@std@@QAEHPBDH@Z):
            00401280   jmp         std::basic_streambuf<char,std::char_traits<char> >::sputn (00403420)
            @ILT+640(?max_size@?$allocator@D@std@@QBEIXZ):
            00401285   jmp         std::allocator<char>::max_size (00405fb0)
            @ILT+645(?_Put@?$num_put@DV?$ostreambuf_iterator@DU?$char_traits@D@std@@@std@@@std@@KA?AV?$ostreambuf_iterator@DU?$char_traits@D@std@@@2@V32@PBDI@Z):
            0040128A   jmp         std::num_put<char,std::ostreambuf_iterator<char,std::char_traits<char> > >::_Put (004052
            @ILT+650(?put@?$num_put@DV?$ostreambuf_iterator@DU?$char_traits@D@std@@@std@@@std@@QBE?AV?$ostreambuf_iterator@DU?$char_traits@D@std@@@2@V32@AAVios_base@2@DK@Z):
            0040128F   jmp         std::num_put<char,std::ostreambuf_iterator<char,std::char_traits<char> > >::put (00401f3
            @ILT+655(?append@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV12@ID@Z):
            00401294   jmp         std::basic_string<char,std::char_traits<char>,std::allocator<char> >::append (00404260)
            @ILT+660(??0sentry@?$basic_ostream@DU?$char_traits@D@std@@@std@@QAE@AAV12@@Z):
            00401299   jmp         std::basic_ostream<char,std::char_traits<char> >::sentry::sentry (00401ca0)
            @ILT+665(??_E?$num_put@DV?$ostreambuf_iterator@DU?$char_traits@D@std@@@std@@@std@@UAEPAXI@Z):
            0040129E   jmp         std::num_put<char,std::ostreambuf_iterator<char,std::char_traits<char> > >::`scalar dele
            @ILT+670(??6std@@YAAAV?$basic_ostream@DU?$char_traits@D@std@@@0@AAV10@PBD@Z):
            004012A3   jmp         std::operator<< (00402bb0)
            @ILT+675(?fill@?$basic_ios@DU?$char_traits@D@std@@@std@@QBEDXZ):
            004012A8   jmp         std::basic_ios<char,std::char_traits<char> >::fill (00401c60)
            @ILT+680(?use_facet@std@@YAABV?$num_put@DV?$ostreambuf_iterator@DU?$char_traits@D@std@@@std@@@1@ABVlocale@1@PBV21@_N@Z):
            004012AD   jmp         std::use_facet (00403020)
            @ILT+685(?flags@ios_base@std@@QBEHXZ):
            004012B2   jmp         std::ios_base::flags (004023b0)
            @ILT+690(?do_grouping@?$numpunct@D@std@@MBE?AV?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@2@XZ):
            004012B7   jmp         std::numpunct<char>::do_grouping (004063c0)
            @ILT+695(?_Grow@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@AAE_NI_N@Z):
            004012BC   jmp         std::basic_string<char,std::char_traits<char>,std::allocator<char> >::_Grow (00404430)
            @ILT+700(?_Ifmt@?$num_put@DV?$ostreambuf_iterator@DU?$char_traits@D@std@@@std@@@std@@KAPADPADDH@Z):
            004012C1   jmp         std::num_put<char,std::ostreambuf_iterator<char,std::char_traits<char> > >::_Ifmt (00404
            @ILT+705(??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAE@PBDABV?$allocator@D@1@@Z):
            004012C6   jmp         std::basic_string<char,std::char_traits<char>,std::allocator<char> >::basic_string<char,
            @ILT+710(??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAE@ABV?$allocator@D@1@@Z):
            004012CB   jmp         std::basic_string<char,std::char_traits<char>,std::allocator<char> >::basic_string<char,
            而且,好像在VC平臺上,虛函數(shù)的指針4字節(jié)放在對象最前面。

            posted on 2007-11-25 22:32 Tim 閱讀(2072) 評論(2)  編輯 收藏 引用 所屬分類: C/C++語言

            評論

            # re: 關(guān)于C++虛函數(shù)的研究 2007-11-26 16:41

            寫程序的時候還是別依虛表表這個東西.
            只要知道多態(tài)就行了.  回復(fù)  更多評論   

            # re: 關(guān)于C++虛函數(shù)的研究 2007-11-30 16:30 jfish

            // 用這個代碼就可以證明虛表實現(xiàn)機(jī)制
            void CallVirtualFun(void* pThis,int index=0){

            void (*funptr)(void*);

            long lVptrAddr;

            memcpy(&lVptrAddr,pThis,4);

            memcpy(&funptr,reinterpret_cast<long*>(lVptrAddr)+index,4);

            funptr(pThis);
            std::cout<<funptr<<std::endl;
            }  回復(fù)  更多評論   

            <2007年7月>
            24252627282930
            1234567
            891011121314
            15161718192021
            22232425262728
            2930311234

            導(dǎo)航

            統(tǒng)計

            公告

            本博客原創(chuàng)文章,歡迎轉(zhuǎn)載和交流。不過請注明以下信息:
            作者:TimWu
            郵箱:timfly@yeah.net
            來源:m.shnenglu.com/Tim
            感謝您對我的支持!

            留言簿(9)

            隨筆分類(173)

            IT

            Life

            搜索

            積分與排名

            最新隨筆

            最新評論

            閱讀排行榜

            久久涩综合| 久久亚洲2019中文字幕| 亚洲AV成人无码久久精品老人| AV无码久久久久不卡蜜桃| 久久国产精品77777| 久久亚洲私人国产精品vA| 99久久99这里只有免费的精品| 久久国产精品国产自线拍免费| 996久久国产精品线观看| 久久99免费视频| 精品久久久久久亚洲| 精品久久久久久无码人妻蜜桃| 久久天天躁狠狠躁夜夜2020| 久久香综合精品久久伊人| 久久久久亚洲AV成人网人人网站| 香蕉99久久国产综合精品宅男自| 99999久久久久久亚洲| 欧美久久综合性欧美| 久久亚洲sm情趣捆绑调教| 91精品国产高清91久久久久久| 人妻精品久久久久中文字幕| 久久精品无码一区二区三区| 午夜精品久久久久| 女人香蕉久久**毛片精品| 亚洲精品乱码久久久久久蜜桃不卡| 久久精品国产亚洲7777| 国产成人综合久久综合| 伊人久久大香线蕉亚洲| 伊人热热久久原色播放www| 一本一道久久精品综合| 91精品国产乱码久久久久久| 久久精品国产2020| 亚洲国产精品一区二区三区久久| www亚洲欲色成人久久精品| 久久久久久久久无码精品亚洲日韩| 人妻无码久久精品| 模特私拍国产精品久久| 精品免费久久久久国产一区| 国产亚洲美女精品久久久| 超级碰久久免费公开视频| 91精品国产91久久久久久蜜臀|