• <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>
            隨筆-5  評論-31  文章-0  trackbacks-0
            題目二:
               題目我做了下改變,使用了上篇文章中提到的那個類X,代碼如下:

             1 class X
             2 {
             3 public:
             4     X(){cout<<"default construct"<<endl;}
             5     X(int a):i(a){ cout<<"construct "<<i<<endl;}
             6     ~X(){ cout<<"desconstruct "<<i<<endl;}
             7     X(const X& x):i(x.i)
             8     {
             9         cout<<"copy construct "<<i<<endl;
            10     }
            11     X& operator++()
            12     {
            13         cout<<"operator ++(pre) "<<i<<endl;
            14         ++i;
            15         return *this;
            16     }
            17     const X operator++(int)
            18     {
            19         cout<<"operator ++(post) "<<i<<endl;
            20         X x(*this);
            21         ++i;
            22         return x;
            23     }
            24     X& operator=(int m)
            25     {
            26         cout<<"operator =(int)"<<endl;
            27         i = m;
            28         return *this;
            29     }
            30     X& operator=(const X& x)
            31     {
            32         cout<<"operator =(X)"<<endl;
            33         i=x.i;
            34         return *this;
            35     }
            36     /////////////////////////
            37     friend ostream& operator<<(ostream& os,const X& x)
            38     {
            39         os<<x.i;
            40         return os;
            41     }
            42     friend X operator+(const X& a,const X& b)
            43     {
            44         cout<<"operator +"<<endl;
            45         return X(a.i+b.i);
            46     }
            47     //////////////////////////
            48 public:
            49     int i;
            50 };

            請問以下代碼的輸出是什么?

            1 X a(10),b(20);
            2 X c=a+b;

            我們來看一下使用GCC4.5(默認編譯選項)以及MSVC9.0(BOTH DEBUG AND RELEASE)編譯后的實際運行結果:
            construct 10
            construct 20
            operator +
            construct 30
            desconstruct 30
            desconstruct 20
            desconstruct 10

            簡單分析下這個輸出:

            construct 10 
            construct 20 //對應 X a(10),b(20);
            operator +  //調用“+”操作符
            construct 30 //調用X(int){...},44行處
            desconstruct 30 //變量c 的析構
            desconstruct 20 //變量b 的析構
            desconstruct 10 //變量a 的析構
             從結果可以看出,整個執行過程中沒有輸出“operator=”,說明壓根沒有調用“=”操作符,而且整個過程比我想象的要簡潔高效,沒有臨時對象,沒有拷貝構造。
            結果為什么會是這樣呢?這主要歸功于編譯器的返回值優化的能力。
            有關返回值優化的知識,限于篇幅我就不仔細介紹了,但是需要特別指出的是MSVC9.0只在RELEASE模式下默認開啟NRVO,即對具名對象的返回值優化,以及返回值優化里面的一個重要的細節,體現在本例里就是:為什么中整個輸出中沒有出現"opeartor=",即為什么沒調用"="操作符。

            現在我們將代碼稍微改變一下,改成下面的樣子:

            X a(10),b(20),c;
            c
            =a+b;  //這里我們將c的構造和賦值分開了

            執行的結果如下:

            construct 10 //構造a
            construct 20 //構造b
            default construct //構造 c
            operator +  //調用“+”操作符
            construct 30 //調用X(int){...},44行處
            operator =(X) //調用“=”操作符
            desconstruct 30 //代碼45行所建立的臨時對象的析構
            desconstruct 30 //變量c的析構
            desconstruct 20 //變量b的析構
            desconstruct 10 //變量c的析構

            對比前后的輸出結果,可以發現多出以下三行
            default construct 
            operator =(X) 
            desconstruct 30 
            出現這種差異的原因在于:
            定義c的時候會調用默認的構造函數進行初始化,因此第一條語句執行完之后,c已經是一個存在的對象,所以第二條語句并沒有權利去直接修改c的內容,必須要通過調用賦值操作符”=“,因此必須要產生一個臨時對象。而在第一個例子中,因為執行到第二條語句之前c并沒有被創建,所以編譯器可以將 表達式a+b的返回值直接構建在c的內存中,從而優化掉臨時對象和對“=”的調用。
            posted on 2011-08-13 21:38 江浸月 閱讀(2103) 評論(7)  編輯 收藏 引用

            評論:
            # re: 做MTK筆試的總結(二)--C++ 返回值優化(RVO) 2011-08-13 22:32 | 瘋狂的面包
            樓主我覺得這個問題就像
            a = (++a) + (a++);
            標準沒有規定 在a++ 執行完之后 a馬上改變吧;只可以保證在這條語句執行完之后,保證a已經改變。  回復  更多評論
              
            # re: 做MTK筆試的總結(二)--C++ 返回值優化(RVO) 2011-08-13 22:48 | 江浸月
            @瘋狂的面包
            沒明白你想表達什么。。能再說明白點么。。  回復  更多評論
              
            # re: 做MTK筆試的總結(二)--C++ 返回值優化(RVO) 2011-08-13 23:18 | 瘋狂的面包
            簡單來說 樓主Google一下 順序點  回復  更多評論
              
            # re: 做MTK筆試的總結(二)--C++ 返回值優化(RVO) 2011-08-13 23:28 | 瘋狂的面包
            給樓主你一個鏈接 http://blog.csdn.net/luciferisnotsatan/article/details/6456696
            自己看好 別糾結  回復  更多評論
              
            # re: 做MTK筆試的總結(二)--C++ 返回值優化(RVO) 2011-08-14 00:03 | 江浸月
            @瘋狂的面包
            首先謝謝你的鏈接,讓我又學到了些。其次,我沒糾結,謝謝。
            這題是一套筆試題里的一個題,對于題目的答案我很不確定,我只是抱著實踐的態度上機試驗,并查閱了相關資料了以后,記錄下心得而已。  回復  更多評論
              
            # re: 做MTK筆試的總結(二)--C++ 返回值優化(RVO) 2011-08-14 19:49 | ybsunok
            X c= a+b;
            編譯器優化 構造函數,復制構造過程,并不是必須的吧。編譯器相關。
              回復  更多評論
              
            # re: 做MTK筆試的總結(二)--C++ 返回值優化(RVO) 2011-08-15 15:57 | 水星家紡
            編譯器優化 構造函數,復制構造過程,并不是必須的吧  回復  更多評論
              
            精品久久久久久久| 国产福利电影一区二区三区久久久久成人精品综合 | 久久人人爽人人爽AV片| 国产亚洲精午夜久久久久久| 品成人欧美大片久久国产欧美... 品成人欧美大片久久国产欧美 | 国产精品综合久久第一页| 伊人色综合久久天天| 久久精品综合一区二区三区| 久久久久久国产精品无码下载| 青草国产精品久久久久久| 国产午夜免费高清久久影院| 久久99热这里只有精品国产| 久久无码AV一区二区三区| 综合网日日天干夜夜久久| 亚洲国产精品久久久久婷婷老年| 久久久免费观成人影院| 久久久一本精品99久久精品66| 久久久久人妻一区精品| 麻豆一区二区99久久久久| 精品国产综合区久久久久久| 人妻精品久久久久中文字幕69| 青青青伊人色综合久久| 亚洲精品乱码久久久久久| 久久精品国产色蜜蜜麻豆| 久久综合精品国产二区无码| 欧美亚洲国产精品久久高清| 久久国产成人| 99久久精品毛片免费播放| 久久久久久久97| 久久无码国产| 国产福利电影一区二区三区久久久久成人精品综合 | 日韩精品久久久肉伦网站| 污污内射久久一区二区欧美日韩 | 久久九九久精品国产免费直播| 久久久精品人妻一区二区三区四| 色播久久人人爽人人爽人人片aV| 国产日产久久高清欧美一区| 久久久老熟女一区二区三区| 精品久久久中文字幕人妻| 思思久久好好热精品国产| 亚洲色欲久久久久综合网|