• <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>
            隨筆-16  評論-116  文章-0  trackbacks-0

            好久沒來了,準備畢業(yè)、畢業(yè)答辯、畢業(yè)、找工作、現(xiàn)在終于穩(wěn)定下來。突然想到一個問題,隨便測試下寫寫,不要拍磚哈

            編譯器:VC2005  Release模式,代碼不優(yōu)化
            調(diào)試器: OllyDBG 1.10
            程序如下:

            class A
            {
            public:
             A();
             
            ~A();
             
            int a;
             
            long b;
            }
            ;
            class B
            {
            public:
             B();
             
            ~B();
             
            int a;
             
            long b;
            }
            ;
            A::A() : a(
            0)
             ,b(
            -1)
            {
            }

            A::
            ~A()
            {
            }

            B::B() 
            {
             a 
            = 1;
             b 
            = -2;
            }

            B::
            ~B()
            {
            }

            int main()
            {
             __asm
             
            {
              push eax   
            //加些標記,方便辨認
              add esp,4
             }

             A a;
             __asm
             
            {
              push ebx   
            //加些標記,方便辨認
              push ebx
              add esp,
            8
             }

             B b;
             
             
            return 0;
            }

            對應的匯編及分析如下:
            對應的匯編及分析如下:
             
            對應的匯編及分析如下:
             
            //類A的構(gòu)造函數(shù)
            00401000  /$  55            push    ebp           //將當前?;穳簵14?/span>
            00401001  |.  8BEC          mov     ebp, esp      //將當前棧指針作為?;?/span>
            00401003  |.  51            push    ecx        //this指針壓棧
            00401004  |.  894D FC       mov     dword ptr [ebp-4], ecx     //保存this指針
            00401007  |.  8B45 FC       mov     eax, dword ptr [ebp-4]     //this指針放入eax
            0040100A  |.  C700 00000000 mov     dword ptr [eax], 0         //this指針指向的地址存入0,也就是變量a,對應語句*(int*)this=0;
            00401010  |.  8B4D FC       mov     ecx, dword ptr [ebp-4]     //this指針放入ecx(多余的一步)
            00401013  |.  C741 04 FFFFF>mov     dword ptr [ecx+4], -1      //this指針指向的地址存入-1,也就是變量b,對應語句*(int*)(this+4)=-1;
            0040101A  |.  8B45 FC       mov     eax, dword ptr [ebp-4]     //返回值放入eax
            0040101D  |.  8BE5          mov     esp, ebp                   //還原先前的esp、ebp
            0040101F  |.  5D            pop     ebp
            00401020  \.  C3            retn
            //中間代碼省略
            //類B的構(gòu)造函數(shù),和a的一模一樣
            00401030  /$  55            push    ebp
            00401031  |.  8BEC          mov     ebp, esp
            00401033  |.  51            push    ecx
            00401034  |.  894D FC       mov     dword ptr [ebp-4], ecx
            00401037  |.  8B45 FC       mov     eax, dword ptr [ebp-4]
            0040103A  
            |.  C700 01000000 mov     dword ptr [eax], 1
            00401040  |.  8B4D FC       mov     ecx, dword ptr [ebp-4]
            00401043  |.  C741 04 FEFFF>mov     dword ptr [ecx+4], -2
            0040104A  
            |.  8B45 FC       mov     eax, dword ptr [ebp-4]
            0040104D  
            |.  8BE5          mov     esp, ebp
            0040104F  
            |.  5D            pop     ebp
            00401050  \.  C3            retn
            //中間代碼省略
            //2個類公用一個析構(gòu)函數(shù)
            00401060  /$  55            push    ebp
            00401061  |.  8BEC          mov     ebp, esp
            00401063  |.  51            push    ecx
            00401064  |.  894D FC       mov     dword ptr [ebp-4], ecx
            00401067  |.  8BE5          mov     esp, ebp
            00401069  |.  5D            pop     ebp
            0040106A  \.  C3            retn
            //中間代碼省略
            //main()函數(shù)
            00401070  /$  55            push    ebp
            00401071  |.  8BEC          mov     ebp, esp
            00401073  |.  83EC 14       sub     esp, 14   //給變量分配20字節(jié)空間
            00401076  |.  50            push    eax      //剛才的代碼標記
            00401077  |.  83C4 04       add     esp, 4   //剛才的代碼標記
            0040107A  |.  8D4D F8       lea     ecx, dword ptr [ebp-8]      //thiscall調(diào)用約定,this指針通過ecx傳遞
            0040107D  |.  E8 7EFFFFFF   call    00401000        //調(diào)用A構(gòu)造函數(shù)
            00401082  |.  53            push    ebx     //剛才的代碼標記
            00401083  |.  53            push    ebx    //剛才的代碼標記
            00401084  |.  83C4 08       add     esp, 8   //剛才的代碼標記
            00401087  |.  8D4D F0       lea     ecx, dword ptr [ebp-10]     //thiscall調(diào)用約定,this指針通過ecx傳遞    
            0040108A  |.  E8 A1FFFFFF   call    00401030   //調(diào)用B構(gòu)造函數(shù)
            0040108F  |.  C745 EC 00000>mov     dword ptr [ebp-14], 0      //函數(shù)返回值預先存儲好
            00401096  |.  8D4D F0       lea     ecx, dword ptr [ebp-10]
            00401099  |.  E8 C2FFFFFF   call    00401060                 //調(diào)用B析構(gòu)函數(shù)
            0040109E  |.  8D4D F8       lea     ecx, dword ptr [ebp-8]
            004010A1  
            |.  E8 BAFFFFFF   call    00401060                 //調(diào)用A析構(gòu)函數(shù)
            004010A6  |.  8B45 EC       mov     eax, dword ptr [ebp-14]   //返回值最后放入eax
            004010A9  |.  8BE5          mov     esp, ebp
            004010AB  
            |.  5D            pop     ebp
            004010AC  \.  C3            retn

            由此可見,成員變量放在初始化列表和構(gòu)造函數(shù)體內(nèi),匯編代碼是一樣的,也就是不管放在那里,是沒有差別的
            posted on 2008-08-07 16:09 greatws 閱讀(2716) 評論(12)  編輯 收藏 引用

            評論:
            # re: 今天做了個小試驗,類成員變量的初始化,發(fā)現(xiàn)放在參數(shù)列表和構(gòu)造函數(shù)體內(nèi)是一樣的 2008-08-07 17:37 | 沈臻豪(foxtail)
            請牛人鑒定一下吧 我記得是不一樣的 一種是初始化 一種是賦值
            不一樣的吧。  回復  更多評論
              
            # re: 今天做了個小試驗,類成員變量的初始化,發(fā)現(xiàn)放在參數(shù)列表和構(gòu)造函數(shù)體內(nèi)是一樣的 2008-08-07 17:40 | 創(chuàng)
            你別用調(diào)試器也可以看出來效果.方法是每個默認構(gòu)造函數(shù)里面加一個打印語句,然后再寫一個有參數(shù)的構(gòu)造函數(shù),也打印語句.看看你把對這些類的構(gòu)造放在函數(shù)體內(nèi)是不是先調(diào)用了默認構(gòu)造函數(shù)再調(diào)用帶參數(shù)的構(gòu)造函數(shù)就行了.
              回復  更多評論
              
            # re: 今天做了個小試驗,類成員變量的初始化,發(fā)現(xiàn)放在參數(shù)列表和構(gòu)造函數(shù)體內(nèi)是一樣的 2008-08-07 18:02 | 過客
            給你一個測試代碼,你就會發(fā)現(xiàn)你是錯誤的:
            using namespace std;


            class A {
            private:
            int aa;
            public:
            A():aa(0) {
            cout << "In A::A()" << endl;
            }
            A(int a):aa(a) {
            cout << "In A::A(int a)" << endl;
            }
            int operator=(int a) {
            cout << "In A::=" << endl;aa = a;
            return a;
            }
            };

            class B {
            private:
            A aa;
            public:
            B() {
            cout << "In B::B()" << endl;
            aa = 3;
            }
            B(int a) : aa(a) {
            cout << "In B::B(int a)" << endl;
            }
            };

            int main(int argc, char **argv, char **env) {
            B b;
            cout << endl;
            B b2(3);
            return 0;
            }
            ================================
            運行結(jié)果:
            In A::A()
            In B::B()
            In A::=

            In A::A(int a)
            In B::B(int a)
              回復  更多評論
              
            # re: 今天做了個小試驗,類成員變量的初始化,發(fā)現(xiàn)放在參數(shù)列表和構(gòu)造函數(shù)體內(nèi)是一樣的 2008-08-07 18:03 | 過客
            為什么是亂的?前面的TAB怎么沒有了?  回復  更多評論
              
            # re: 今天做了個小試驗,類成員變量的初始化,發(fā)現(xiàn)放在參數(shù)列表和構(gòu)造函數(shù)體內(nèi)是一樣的 2008-08-07 18:04 | 過客
            [CODE]
            using namespace std;


            class A {
            private:
            int aa;
            public:
            A():aa(0) {
            cout << "In A::A()" << endl;
            }
            A(int a):aa(a) {
            cout << "In A::A(int a)" << endl;
            }
            int operator=(int a) {
            cout << "In A::=" << endl;aa = a;
            return a;
            }
            };

            class B {
            private:
            A aa;
            public:
            B() {
            cout << "In B::B()" << endl;
            aa = 3;
            }
            B(int a) : aa(a) {
            cout << "In B::B(int a)" << endl;
            }
            };

            int main(int argc, char **argv, char **env) {
            B b;
            cout << endl;
            B b2(3);
            return 0;
            }


            [/CODE]  回復  更多評論
              
            # re: 今天做了個小試驗,類成員變量的初始化,發(fā)現(xiàn)放在參數(shù)列表和構(gòu)造函數(shù)體內(nèi)是一樣的 2008-08-07 18:10 | 過客
            不弄了,將就看吧,呵呵

            你會發(fā)現(xiàn)構(gòu)造的過程是:要先構(gòu)造類成員變量,然后才會執(zhí)行構(gòu)造函數(shù)體里面的代碼。

            如果你把賦值代碼放在函數(shù)體里,有關的數(shù)據(jù)被賦了兩次值。

            而基本數(shù)據(jù)類型則不需要第一次構(gòu)造的過程,所以你認為過程是一樣的

            ~~~~拍磚~~~~  回復  更多評論
              
            # re: 今天做了個小試驗,類成員變量的初始化,發(fā)現(xiàn)放在參數(shù)列表和構(gòu)造函數(shù)體內(nèi)是一樣的 2008-08-07 18:25 | 過客
            再解釋一下哦:

            在編譯之后,編譯器會在你的構(gòu)造函數(shù)的最前面依次加上各成員變量的默認構(gòu)造函數(shù),而如果你在構(gòu)造名稱后指定了某些成員變量的構(gòu)造參數(shù)的話,相關的項目就會被你指定的參數(shù)所代替。

            在你的代碼中看不出這種區(qū)別,因為int不需要構(gòu)造函數(shù)(這個是C的原因,有的語言會把int初始化成0),只是在你指定時,編譯器在最前面加入賦值語句。
              回復  更多評論
              
            # re: 今天做了個小試驗,類成員變量的初始化,發(fā)現(xiàn)放在參數(shù)列表和構(gòu)造函數(shù)體內(nèi)是一樣的 2008-08-07 18:27 | 過客
            同拍
            樓上正解
            因為你的是基本類型不存在構(gòu)造函數(shù)  回復  更多評論
              
            # re: 今天做了個小試驗,類成員變量的初始化,發(fā)現(xiàn)放在參數(shù)列表和構(gòu)造函數(shù)體內(nèi)是一樣的 2008-08-07 20:11 | greatws
            謝謝各位高手的指正,是我疏忽了,只考慮了一種情況  回復  更多評論
              
            # re: 今天做了個小試驗,類成員變量的初始化,發(fā)現(xiàn)放在參數(shù)列表和構(gòu)造函數(shù)體內(nèi)是一樣的 2008-08-07 22:06 | lonkil
            這樣的討論方式,不錯,力頂。  回復  更多評論
              
            # re: 今天做了個小試驗,類成員變量的初始化,發(fā)現(xiàn)放在參數(shù)列表和構(gòu)造函數(shù)體內(nèi)是一樣的 2008-08-07 23:40 | winsty
            構(gòu)造函數(shù)和拷貝構(gòu)造必然不同的吧……
            只是因為你實驗的是int這種類型……  回復  更多評論
              
            # re: 今天做了個小試驗,類成員變量的初始化,發(fā)現(xiàn)放在參數(shù)列表和構(gòu)造函數(shù)體內(nèi)是一樣的[未登錄] 2008-08-11 10:05 | raof01
            怎么可能一樣?
            你試試成員是其他類對象的情況看看。如A和B兩個類示范的。  回復  更多評論
              
            少妇久久久久久被弄到高潮 | 久久精品亚洲欧美日韩久久| 久久99精品国产麻豆宅宅 | 国产精品久久久久影院色| 国产精品天天影视久久综合网| 91麻豆精品国产91久久久久久| 久久亚洲国产精品五月天婷| 亚洲中文字幕久久精品无码APP | 久久久久国产一级毛片高清板| 久久天天躁狠狠躁夜夜avapp | 久久久无码精品午夜| 久久大香香蕉国产| 久久中文字幕无码专区| 久久精品人成免费| 欧美与黑人午夜性猛交久久久 | 国产精品久久久久久久久免费 | 久久久久亚洲爆乳少妇无| 久久人人爽爽爽人久久久| 久久久免费观成人影院| 久久精品www| 久久水蜜桃亚洲av无码精品麻豆| 欧美精品福利视频一区二区三区久久久精品 | 国产精品无码久久综合 | 99久久精品免费看国产免费| 午夜天堂av天堂久久久| 亚洲欧洲久久久精品| 精品无码久久久久久国产| 91精品国产91久久综合| 久久久久久亚洲AV无码专区| 免费精品久久天干天干| 深夜久久AAAAA级毛片免费看| 精品国产综合区久久久久久| 91麻精品国产91久久久久| 久久这里只精品国产99热| 精品熟女少妇a∨免费久久| 久久久久久久久无码精品亚洲日韩 | 一本久道久久综合狠狠爱| 热99RE久久精品这里都是精品免费 | 狠狠色丁香婷婷综合久久来来去| 久久91精品国产91久久户| 99久久精品国产一区二区三区|