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

            brent's hut

            C++拷貝構造函數深入分析以及重寫operator =

            ?

            class ?CTestCopyConstruct {
            public :
            ????CTestCopyConstruct()
            {
            ????????TRACE(
            " Enter?CTestCopyConstruct();this?is?%d\n " , this );
            ????????strTest?
            = ? " not?ok " ;
            ????????i?
            = ? 0 ;
            ????}

            ????CTestCopyConstruct(
            const ?CTestCopyConstruct? & src) {
            ????????TRACE(
            " Enter?CTestCopyConstruct(const?CTestCopyConstruct?&src);this?is?%d;src?is?%d\n " , this , & src);
            ????????strTest?
            = ?src.strTest;
            ????????i?
            = ?src.i;
            ????}

            ????CTestCopyConstruct?
            & ?operator? = ( const ?CTestCopyConstruct? & ?src) {
            ????????TRACE(
            " Enter?CTestCopyConstruct?&?operator?=(const?CTestCopyConstruct?&?src);this?is?%d;src?is?%d\n " , this , & src);
            ????????strTest?
            = ?src.strTest;
            ????????i?
            = ?src.i;
            ????????
            return ? * this ;
            ????}

            ????CString?strTest;
            ????
            int ?i;
            }
            ;

            CTestCopyConstruct?GetTest()
            {
            ????CTestCopyConstruct?ret1;
            ????ret1.strTest?
            = ? " ok " ;
            ????ret1.i?
            = ? 0 ;
            ????CTestCopyConstruct?ret2;

            ????
            return ?ret1;
            }


            void ?CTestDlg::OnOK()?
            {
            ????CTestCopyConstruct?var1;
            ????CTestCopyConstruct?var2?
            = ?GetTest();

            ????TRACE(
            " \nresult?1:\n " );
            ????TRACE(
            " var1?is?%d\n " , & var1);
            ????TRACE(
            " var2?is?%d?var2.str?is?%s\n\n " , & var2,var2.strTest);

            ????CTestCopyConstruct?var3?
            = ?var2;
            ????CTestCopyConstruct?var4;
            ????var4?
            = ?var2;

            ????TRACE(
            " \nresult?2:\n " );
            ????TRACE(
            " var3?is?%d?var3.str?is?%s\n " , & var3,var3.strTest);
            ????TRACE(
            " var4?is?%d?var2.str?is?%s\n " , & var4,var4.strTest);
            }



            代碼如上,調試窗口輸出如下:
            Enter CTestCopyConstruct();this is 1242980
            Enter CTestCopyConstruct();this is 1242848
            Enter CTestCopyConstruct();this is 1242840
            Enter CTestCopyConstruct(const CTestCopyConstruct &src);this is 1242972;src is 1242848

            result 1:
            var1 is 1242980
            var2 is 1242972 var2.str is ok

            Enter CTestCopyConstruct(const CTestCopyConstruct &src);this is 1242964;src is 1242972
            Enter CTestCopyConstruct();this is 1242956
            Enter CTestCopyConstruct & operator =(const CTestCopyConstruct & src);this is 1242956;src is 1242972

            result 2:
            var3 is 1242964 var3.str is ok
            var4 is 1242956 var2.str is ok

            分析:
            CTestCopyConstruct var1;\\1
            CTestCopyConstruct var2 = GetTest();\\2
            代碼的執行如下:
            當前堆棧指針(sp) = 1242980
            sp -= 8//在堆棧中為var1分配空間
            在var1上(1242980 - 1242973)調用構造函數
            sp -= 8//在堆棧中為var2分配空間
            sp -= n//保護當前環境
            進入了GetTest函數
            當前sp = 1242848
            sp -= 8//為ret1分配空間
            構建ret1
            sp -= 8//為ret2分配空間
            構建ret2
            ......
            對var2(1242972處的堆棧段)調用拷貝構造函數,以test1(1242848處)為參數
            //析構test1 test2等...
            sp += n//恢復運行環境
            ......

            另:
            operater = () 和默認構造函數不一樣,只重寫=運算符而不提供拷貝構造函數,調用的仍然是默認的構造函數。
            默認構造函數和賦值運算符處理的情況不一樣,一個是在已分配的空間上調用,一個是在已構造的對象上調用。

            默認拷貝構造函數會調用類中各成員的拷貝構造函數。CString 由于提供了拷貝構造函數,所以上面例子中即使去掉拷貝構造函數,var2 仍然會得到正確的值。

            調試的環境是vc6.0 debug 默認選項。編譯沒有優化。

            CTestCopyConstruct( const ?CTestCopyConstruct? & src)
            ????????
            {
            ????????TRACE(
            " Enter?CTestCopyConstruct(const?CTestCopyConstruct?&src);this?is?%d;src?is?%d\n " , this , & src);
            ????????strTest?
            = ?src.strTest;
            ????????i?
            = ?src.i;
            ????}




            CTestCopyConstruct(
            const ?CTestCopyConstruct? & src)
            ????????:strTest?(src.strTest)
            {
            ????????TRACE(
            " Enter?CTestCopyConstruct(const?CTestCopyConstruct?&src);this?is?%d;src?is?%d\n " , this , & src);
            ????????i?
            = ?src.i;
            ????}

            前者先調用了CString::CString()再調用CString::operator =
            后者直接調用了CString::CString(CString & src);


            默認的賦值運算的行為:首先調用父類的賦值運算。
            然后會為自己獨有的各成員尋找賦值運算。如果成員的賦值運算符被重寫,則調用這個重寫的賦值運算符函數,如果這個重寫的運算符函數是private,編譯將無法通過。
            默認的拷貝構造函數的行為:首先調用父類的拷貝構造函數。
            然后為自己獨有的各成員尋找拷貝構造函數。如果這個成員提供拷貝構造函數,則調用之,如果成員的類提供的拷貝構造函數是private,編譯將無法通過。
            (子類完全可以把父類當成自己的一個成員?)


            可以說默認的賦值運算和默認的拷貝構造函數是類最常被用到的兩個函數了...內部卻不是一般的復雜。

            posted on 2006-03-30 10:34 brent 閱讀(1594) 評論(0)  編輯 收藏 引用 所屬分類: C++

            欧美麻豆久久久久久中文| 国产成人精品白浆久久69| 亚洲精品乱码久久久久久不卡| 久久久久无码专区亚洲av| 无码人妻久久一区二区三区蜜桃 | 欧美国产成人久久精品| 人妻精品久久无码区| 九九热久久免费视频| 色综合久久久久综合体桃花网 | 66精品综合久久久久久久| 亚洲欧美日韩久久精品| 久久91精品国产91久久麻豆| 欧美日韩中文字幕久久久不卡| 国内精品久久久久伊人av| 亚洲一区精品伊人久久伊人| 欧美精品一区二区精品久久| 99久久国产宗和精品1上映| 久久精品国产亚洲5555| 欧美激情精品久久久久| 蜜臀av性久久久久蜜臀aⅴ麻豆 | 久久综合香蕉国产蜜臀AV| 久久成人18免费网站| 国产韩国精品一区二区三区久久| 青青草原综合久久大伊人| 亚洲色欲久久久久综合网| 国产精品欧美久久久久天天影视 | 久久国产V一级毛多内射| 国产午夜免费高清久久影院| 伊人久久大香线蕉亚洲| 久久精品综合网| 狠狠精品久久久无码中文字幕| 色偷偷91久久综合噜噜噜噜| 丰满少妇人妻久久久久久4| 岛国搬运www久久| 人妻少妇精品久久| 亚洲午夜精品久久久久久app| 亚洲人AV永久一区二区三区久久| 久久综合精品国产一区二区三区| 久久青青草原精品国产软件| 国产一区二区精品久久岳| 久久精品这里只有精99品|