• <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>
            posts - 200, comments - 8, trackbacks - 0, articles - 0

            (i++)+(i++)與(++i)+(++i) (轉)

            Posted on 2012-11-19 20:16 鑫龍 閱讀(246) 評論(0)  編輯 收藏 引用 所屬分類: C

            與在前面:++(--)有太多讓人困惑的地方,(i++)+(i++)與(++i)+(++i)有什么不同?為什么不同?如果從機器的角度去理解,就會豁然開朗。

             先來看段程序:

            復制代碼
            int main()
            {
                
            int i=3;
                
            int j=(i++)+(i++);
                
            //    int j=(++i)+(++i);
                printf("%d,%d\n",i,j);
            }
            復制代碼

            (1)在VC 6.0下:

            對于(i++)+(i++):
            結果:i=5,j=6

            相應的匯編代碼為(有詳細注釋):

            復制代碼
            8B 45 FC             mov         eax,dword ptr [ebp-4]   ;i->eax
            03 45 FC             add         eax,dword ptr [ebp-4]    ;i+i=6
            89 45 F8             mov         dword ptr [ebp-8],eax    ;6->j
            8B 4D FC             mov         ecx,dword ptr [ebp
            -4]    ;i->ecx(=3)
            83 C1 01             add         ecx,1                           ;ecx=4
            89 4D FC             mov         dword ptr [ebp-4],ecx    ;4->i
            8B 
            55 FC             mov         edx,dword ptr [ebp-4]    ;i->edx
            83 C2 01             add         edx,1                           ;edx=5
            89 55 FC             mov         dword ptr [ebp-4],edx    ;5->i
            復制代碼

             
            對于(++i)+(++i):
            結果:i=5,j=10
            相應的匯編代碼為:

            復制代碼
            8B 45 FC             mov         eax,dword ptr [ebp-4]    ;i->eax (=3)
            83 C0 01             add         eax,1                           ;eax=4
            89 45 FC             mov         dword ptr [ebp-4],eax    ;4->i
            8B 4D FC             mov         ecx,dword ptr [ebp
            -4]    ;i->ecx
            83 C1 01             add         ecx,1                           ;ecx=5
            89 4D FC             mov         dword ptr [ebp-4],ecx    ;5->i
            8B 
            55 FC             mov         edx,dword ptr [ebp-4]    ;i->edx
            03 55 FC             add         edx,dword ptr [ebp-4]    ;edx=10 ,即i+i
            89 55 F8             mov         dword ptr [ebp-8],edx    ;10->j
            復制代碼

             
            (2)在gcc 3.2.2下:

            對于(i++)+(i++):

            結果:i=5,j=6相應的匯編代碼為:

            復制代碼
            c7 45 fc 03 00 00 00     movl    $3, -4(%ebp)        ;3->i
            8b 55 fc        movl    -4(%ebp), %edx        ;i->edx (=3)
            8b 45 fc        movl    -4(%ebp), %eax        ;i->eax    (=3)
            8d 04 10         leal    (%eax,%edx), %eax     ;i+i=6 ->eax
            89 45 f8        movl    %eax, -8(%ebp)        ;6->j
            8d 45 fc        leal    -4(%ebp), %eax        ;&i->eax
            ff 00            incl    (%eax)            ;i++ ,即i=4,注意這里為寄存器間接尋址
            8d 45 fc        leal    -4(%ebp), %eax        ;&i->eax
            ff 00            incl    (%eax)                ;i++,即i=5
            復制代碼

             
            對于(++i)+(++i):
            結果:i=5,j=10
            相應的匯編代碼為:

            復制代碼
            movl    $3, -4(%ebp)        ;3->i
            leal    -4(%ebp), %eax        ;&i->eax
            incl    (%eax)            ;i++,即i=4
            leal    -4(%ebp), %eax        ;&i->eax
            incl    (%eax)            ;i++, i=5
            movl    -4(%ebp), %eax        ;i->eax, eax=5
            addl    -4(%ebp), %eax        ;i+i ->eax ,eax=10
            movl    %eax, -8(%ebp)        ;10->j
            復制代碼


            可見,對于VC6.0和gcc,二者的結果一致,但是gcc 3.2.2生成的匯編代碼明顯比VC6.0高效、簡潔。這也許是因為VC 6.0出現較早的原因吧。

             

            (3)如果這段代碼用java實現,結果會怎樣呢?

            程序:

            復制代碼
            public class TestAdd {
                
            public static void main(String[] args) {
                    
            int i=3;
                    
            int j=(i++)+(i++);    //5,7
                    
            //int j=(++i)+(++i);  //5,9
                    System.out.println(i+","+j);
                }
            }
            復制代碼

             對于(++i)+(++i):
            i=5,j=9。結果點意外!
            來看看它的字節碼吧:

            復制代碼
            //j=(++i)+(++i)
            //
            5,9
             
            0:   iconst_3            ;常量3入棧
             1:   istore_1            ;從棧中彈出3,存入i,i=3
             2:   iinc    11         ;i++, i=4
             5:   iload_1              ;將i壓入棧,即4入棧
             6:   iinc    11          ; i++,i=5
             9:   iload_1               ;i入棧,即5入棧
             10:  iadd                  ;從棧中彈出兩個int類型的數相加,結果入棧,即9入棧
             11:  istore_2            ;從棧中彈出9,存入j,即j=9
            復制代碼

             對于(i++)+(i++):
            i=5,j=7。結果也很意外!
            也來看看它的字節碼吧:

            復制代碼
            //j=(i++)+(i++)
            //
            5,7
             
            0:   iconst_3            ;常量3入棧
             1:   istore_1            ;從棧中彈出3,存入i,i=3
             2:   iload_1               ;i入棧,即3入棧
             3:   iinc    11          ;i++,即i=4
             6:   iload_1               ;i入棧,即4入棧
             7:   iinc    11           ;i++,即i=5;注意:5沒有入棧,所以此時棧中的數為3和4
             10:  iadd                  ;從棧彈出兩個int類型數相加,結果入棧,即7入棧
             11:  istore_2            ;從棧中彈出7,存入j,即j=7
            復制代碼

             

            Java與VC/gcc為什么會有如此的區別呢?其實原因很簡單,VC/gcc生成的是本地代碼,而X86處理器是基于寄存器的架構,也就是如果它要進行了兩個數相加,它會先把兩個數移到寄存器,再進行加法運算。而Java虛擬機是一種基于棧的架構,如果它要進行兩個數相加,它會先彈出兩個數,再進行加法運算,再將結果入棧。

            麻豆一区二区99久久久久| 尹人香蕉久久99天天拍| 久久精品成人国产午夜| 国产精品久久永久免费| 国产亚洲精久久久久久无码| 亚洲狠狠久久综合一区77777| 久久久国产精华液| 久久久久无码精品国产不卡| 日本精品久久久中文字幕| 精品久久久久久久久免费影院| 亚洲人成伊人成综合网久久久| 久久精品国产影库免费看| 欧美国产精品久久高清| 国产精品久久久福利| 久久久久国产精品嫩草影院 | 久久久久亚洲AV成人网| 伊人久久大香线蕉av不卡 | 久久国产乱子伦精品免费强| 久久精品亚洲福利| 久久国产精品99久久久久久老狼| 亚洲人成无码久久电影网站| 91久久福利国产成人精品| 久久久久久毛片免费播放| 大香伊人久久精品一区二区| 欧美精品丝袜久久久中文字幕 | 精品久久久久久中文字幕| 无码伊人66久久大杳蕉网站谷歌| 久久天天躁狠狠躁夜夜2020| 久久久久一区二区三区| 国产精品9999久久久久| 99久久99久久精品免费看蜜桃| 久久精品视频一| 久久久亚洲裙底偷窥综合| 国产精品亚洲综合久久| 亚洲精品国产第一综合99久久| 久久精品成人影院| 天天做夜夜做久久做狠狠| 亚洲精品无码久久久| 久久一区二区三区免费| 久久久亚洲裙底偷窥综合| 亚洲AV日韩精品久久久久|