• <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ù)膶?shí)現(xiàn)
            豪邁的氣魄、頑強(qiáng)的意志和周全的思考

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

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

            #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個(gè)字節(jié)外,還有4個(gè)字節(jié)多于。最終調(diào)試發(fā)現(xiàn),只要父類有虛函數(shù),而又無論函數(shù)個(gè)數(shù)是多少,父類和所有繼承于他的子類都多4個(gè)字節(jié),做什么用的呢?存放一個(gè)指針,指向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平臺(tái)上,虛函數(shù)的指針4字節(jié)放在對(duì)象最前面。

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

            評(píng)論

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

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

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

            // 用這個(gè)代碼就可以證明虛表實(shí)現(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ù)  更多評(píng)論   

            <2012年4月>
            25262728293031
            1234567
            891011121314
            15161718192021
            22232425262728
            293012345

            導(dǎo)航

            統(tǒng)計(jì)

            公告

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

            留言簿(9)

            隨筆分類(173)

            IT

            Life

            搜索

            積分與排名

            最新隨筆

            最新評(píng)論

            閱讀排行榜

            久久久久无码精品国产不卡| 精品欧美一区二区三区久久久 | 国产精品99久久久精品无码| 亚洲精品国精品久久99热| 亚洲午夜福利精品久久| 亚洲精品乱码久久久久久蜜桃不卡 | 大蕉久久伊人中文字幕| 婷婷久久综合九色综合九七| 精品久久人妻av中文字幕| 热久久国产精品| 久久99精品国产麻豆宅宅| 国产一级持黄大片99久久| 久久精品国产亚洲AV影院| 国产精品青草久久久久福利99 | 久久亚洲高清观看| 久久综合久久美利坚合众国 | 亚洲va国产va天堂va久久| 天天爽天天爽天天片a久久网| 久久精品人人做人人爽电影| 国产高潮国产高潮久久久91 | 国内精品久久久久影院薰衣草| 伊人久久大香线蕉影院95| 狠狠88综合久久久久综合网| 亚洲精品蜜桃久久久久久| 亚洲?V乱码久久精品蜜桃| 狠狠色伊人久久精品综合网| 日本免费一区二区久久人人澡| 国产亚洲欧美精品久久久| 久久久久精品国产亚洲AV无码| 四虎影视久久久免费| 久久成人永久免费播放| 国产精品xxxx国产喷水亚洲国产精品无码久久一区 | 久久亚洲中文字幕精品一区四| 99久久精品午夜一区二区| 99久久超碰中文字幕伊人| 久久综合给合久久狠狠狠97色69| 色天使久久综合网天天| 久久久久久精品成人免费图片| 久久久久久曰本AV免费免费| 久久精品卫校国产小美女| 中文字幕日本人妻久久久免费|