• <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>

            zssure

              C++博客 :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
              0 隨筆 :: 2 文章 :: 0 評論 :: 0 Trackbacks

            MSDN中對VS2012版本的臨時對象的說明如下:

                在某些情況下,編譯器有必要產生臨時對象。

            1.     當初始化一個常量引用const reference)時,如果給定的初始化對象類型與目標引用類型不同(但是兩者  能夠相互轉換),需要產生臨時對象;
            2.     當函數(shù)的返回值是用戶自定義類型,且程序中未將此返回值拷貝到其他對象中時,需要產生臨時對象;
            3.     當給定的對象顯式向自定義對象類型轉換時,產生臨時對象;

            IBM官網上的給出的描述如下:

                C++中編譯器有些時候有必要產生臨時對象。通常在初始化引用、計算(評估evaluation)含有標磚類型轉換的表達式、參數(shù)傳遞、函數(shù)返回、評估異常拋出表達式(throw expression)。

            參考資料:

            http://msdn.microsoft.com/en-us/library/a8kfxa78(v=vs.110).aspx 

            http://publib.boulder.ibm.com/infocenter/comphelp/v7v91/index.jsp?topic=%2Fcom.ibm.vacpp7a.doc%2Flanguage%2Fref%2Fclrc03cplr116.htm 

                總之,讀完之后,是不是還感覺臨時對象捉摸不定呢?的確,C++標準中沒有明確給出臨時對象的產生規(guī)則和條件,由編譯器自動產生的,不管是處于效率還是其他原因,各編譯器之間的產生時機和方式都略有不同,下面就MSVC編譯器進行一些基本的探討,下面的代碼都是在VS2008環(huán)境下編譯通過的。

            情況一:

                通過不同設數(shù)據(jù)類型來初始化常量引用

                在(1)代碼處,設置斷點,進行調試,查看反匯編如下:


                其中fild dword ptr[ebp-8] 是將整型變量iNum轉換成浮點型(float)并壓棧(相當于復制了一份iNum的數(shù)據(jù)存儲到浮點寄存器中)?!居嘘P浮點數(shù)的匯編指令,可參見百度百科相關說明】。然后fstp dwrod ptr[ebp-20h]是出棧指令,將剛才存儲的浮點數(shù)據(jù)轉存到ebp-20h??臻g中。也就是說ebp-20h處就是我們要找的“臨時變量”。通過對比const float& r10=fNum;一行的反匯編指令,結果就更明顯了。

            情況二:當函數(shù)返回值是自定義類型時。

            我們知道,通常函數(shù)的返回值有兩種方式來傳遞:寄存器和棧。內置數(shù)據(jù)類型,通常都是通過寄存器(MSVC下是EAX)來直接作為返回值的存儲容器,將結果由被調用函數(shù)傳遞給調用者。但是如果返回值是字符串、數(shù)組等比較大的數(shù)據(jù)結構時,一個32位的寄存器EAX是不夠的,此時常常會利用到其他的寄存器(例如ECX、EDX)等來進行轉存容器。自定義的類對象,通常通過在棧中開辟一塊空間(大小由編譯器根據(jù)類對象的大小自動設定)來轉存返回值。所以可以簡單的認為在棧中開辟的轉存空間就是一份返回值的副本,即“臨時對象”。通過寄存器的轉存,我們不能說是“臨時對象”,因為CPU的一切數(shù)據(jù)操作都是通過各種寄存器來完成的,況且寄存器中的數(shù)據(jù)也不存在聲明周期問題,隨時有可能被覆蓋掉,當然如果進行了進棧操作,那就可以叫做臨時對象了。

            示例代碼如下:

            main()函數(shù)中測試代碼:

                                             
            圖一

            Point是自定義的一個二維平面點類(因為如果直接定義成簡單的POD類,編譯器會直接將類進行優(yōu)化,當做兩個單獨的整型數(shù)據(jù)成員來看,所以在定義類時,添加了自己的構造函數(shù)、析構函數(shù)、operator +、取值和設定函數(shù))。類結構如下:


                                         圖二

            程序的直接運行(CTRL+F5)結果如下:


            由輸出結果可以直接看出,構造函數(shù)與析構函數(shù)不對稱,其中0012FE54處的對象只調用了析構函數(shù),但是沒有調用構造函數(shù)。所以可以懷疑0012FE54應該是一個“temporary object”。(但是為什么可以不調用構造函數(shù),卻要調用析構函數(shù)呢?這一點我還沒搞明白,有待考究一下)。

            下面單步調試進入到函數(shù)體中,看看究竟:

            首先進入到(圖一)中的(1)處:

            其反匯編代碼如下:

            其中,004111C2是構造函數(shù)Point(int x, int y)在跳轉表中的位置,如下:


            如圖所示,再次跳轉到Pointint x, int y)構造函數(shù)00411720處。

            Pointint x, int y)構造函數(shù)反匯編代碼如下:

            可以用下面的示意圖來表示有參構造函數(shù)的操作過程:

            同樣,運行到(圖一)中(2)時,結果完全相同:

            示意圖如下:


            然后到(圖一)中的(3)處:

            此時調用的是無參構造函數(shù),其反匯編代碼如下:

            操作過程示意圖如下;


            最后,到了我們此部分的重點,p3=p1+p2;

            其反匯編代碼如下:


            可以看得出來,在調用operator+00411113)函數(shù)時,壓入了三個參數(shù)進棧:分別是p1的地址、p2的地址,和0x0012FF54(——我們所找的臨時對象)。下面就進入到了oeprator+友元函數(shù)體中,其反匯編代碼如下:

            從反匯編代碼可以看出,在return p3;語句時,將局部對象p30x0012FE10)拷貝到臨時對象0x0012FE54(進入函數(shù)前壓入棧中的第三個參數(shù)——我們要找的臨時對象)中。

            拷貝完成后,局部變量p3(0x0012FE10)調用其析構函數(shù)"call 00411037"。此時,由于p3=p1+p2求值表達式(evaluation expression)還未完成,所以臨時對象(0012FE54)并未調用其析構函數(shù)。

            接著往下看。


            然后將臨時對象中的值拷貝到了主函數(shù)的局部對象p3中(如圖中紅色代碼所示)。至此,求值表達式運算完成,意味著臨時對象(0x0012FE54)的生存已失去意義,遂將0x0012FE54復制到ECX中壓棧,然后調用析構函數(shù)。

            函數(shù)最后:return 0;時,主函數(shù)中的p1,p2,p3生命周期也就結束了,所以按照與構造函數(shù)相反的順序依次調用其析構函數(shù),反匯編代碼如下:


            參考書籍:

            《深度探索C++對象模型》

            C++反匯編與逆向分析技術揭秘》
            注:文中是個人學習過程中的筆記,歡迎大家批評指正,交流溝通才能進步。

            個人郵箱:zssure@163.com

            posted on 2012-07-01 19:24 zssure 閱讀(172) 評論(0)  編輯 收藏 引用
            精品久久人人爽天天玩人人妻| 久久婷婷五月综合成人D啪| 亚洲国产日韩欧美综合久久| 亚洲伊人久久综合影院| 奇米影视7777久久精品人人爽| 亚洲欧美一区二区三区久久| 久久久久精品国产亚洲AV无码| 欧美日韩精品久久免费| 91精品国产色综合久久| 国产精品成人99久久久久91gav| 久久综合伊人77777| 久久久久久久久无码精品亚洲日韩 | 亚洲AV无码久久| 91性高湖久久久久| 99精品国产综合久久久久五月天| 精品免费久久久久久久| 免费精品久久久久久中文字幕| 男女久久久国产一区二区三区| 国产免费久久精品99re丫y| 99久久久精品免费观看国产| 久久久久国产精品嫩草影院| 久久久久亚洲AV无码麻豆| 久久伊人亚洲AV无码网站| www.久久精品| 久久久久久精品无码人妻| 久久久久久无码国产精品中文字幕| 一本色道久久88—综合亚洲精品| 国产99久久久国产精免费| 1000部精品久久久久久久久| 国产69精品久久久久APP下载| 99久久国产综合精品成人影院| 色欲av伊人久久大香线蕉影院| 久久精品国产亚洲av瑜伽| 久久久久久久99精品免费观看| 久久精品国产久精国产果冻传媒 | 欧美精品一区二区精品久久| 色诱久久久久综合网ywww| 色综合久久久久综合体桃花网 | 蜜臀av性久久久久蜜臀aⅴ| 无码久久精品国产亚洲Av影片| 午夜精品久久久久|