• <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 江浸月 閱讀(2098) 評論(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 | 水星家紡
            編譯器優化 構造函數,復制構造過程,并不是必須的吧  回復  更多評論
              
            热re99久久6国产精品免费| 一级做a爰片久久毛片免费陪| 精品无码久久久久久久动漫| 久久天天躁狠狠躁夜夜2020| 久久人妻少妇嫩草AV无码专区| 久久久久久国产a免费观看黄色大片| 久久精品无码一区二区app| 国产精品99久久精品爆乳| 国产成人精品久久综合 | …久久精品99久久香蕉国产| 精品久久久久久无码不卡| 一级做a爰片久久毛片看看| 久久久久av无码免费网| 亚洲国产欧洲综合997久久| 久久亚洲AV成人无码国产| 国产精品久久久久久吹潮| 久久福利青草精品资源站免费| 久久夜色精品国产www| 偷窥少妇久久久久久久久| 久久这里只有精品18| 91亚洲国产成人久久精品| 国产精品免费久久久久影院| 国产午夜精品久久久久九九电影| 久久精品无码一区二区三区日韩| 2020久久精品亚洲热综合一本| 久久久av波多野一区二区| 久久996热精品xxxx| 婷婷伊人久久大香线蕉AV| 久久se精品一区二区| 久久久久国产精品嫩草影院| 婷婷综合久久中文字幕蜜桃三电影 | 蜜臀久久99精品久久久久久小说| 999久久久免费精品国产| 日本久久久久久久久久| 999久久久免费精品国产| 四虎久久影院| 久久久久久久综合日本| 精品国产乱码久久久久久郑州公司| 久久久精品日本一区二区三区| 久久精品国产亚洲AV高清热 | av国内精品久久久久影院|