• <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| 久久精品无码一区二区日韩AV| 色8激情欧美成人久久综合电| 无码人妻久久一区二区三区| av无码久久久久不卡免费网站| 久久成人精品| 日本久久久久亚洲中字幕| 亚洲国产精品久久久久| 亚洲综合精品香蕉久久网| 精品久久久久久久| 中文字幕日本人妻久久久免费| 国产精品99久久精品爆乳| 无码超乳爆乳中文字幕久久 | 区久久AAA片69亚洲| 青青青国产成人久久111网站| 欧美久久久久久| 久久精品人妻一区二区三区| 久久精品国产99久久久| 中文精品久久久久人妻| 久久99热这里只频精品6| 香蕉久久夜色精品国产小说| 777午夜精品久久av蜜臀| 久久久久久精品免费免费自慰| 99久久国产亚洲高清观看2024| 久久精品国产只有精品2020| 久久精品无码专区免费东京热| 天天躁日日躁狠狠久久| 精品熟女少妇AV免费久久| 久久人人爽人人爽AV片| 精品久久久久久久中文字幕 | 国产免费久久精品丫丫| 久久精品9988| 26uuu久久五月天| 国产高潮国产高潮久久久91| 老司机国内精品久久久久| 色综合久久综精品| 国产福利电影一区二区三区,免费久久久久久久精 | 久久精品亚洲中文字幕无码麻豆| 亚洲中文久久精品无码|