資源描述:
《如何理解拷貝構(gòu)造函數(shù)》由會員上傳分享,免費在線閱讀,更多相關(guān)內(nèi)容在應(yīng)用文檔-天天文庫。
1、我的心得:對結(jié)構(gòu)和對象的參數(shù)傳遞和函數(shù)返回,盡量用引用,不要用值傳遞,一是可以節(jié)省內(nèi)存開銷,二是避免再創(chuàng)造臨時對象或結(jié)構(gòu)時,引起錯誤(特別是包含動態(tài)分配成員時)拷貝構(gòu)造函數(shù)??拷貝構(gòu)造函數(shù),是一種特殊的構(gòu)造函數(shù),它由編譯器調(diào)用來完成一些基于同一類的其他對象的構(gòu)建及初始化。其唯一的參數(shù)(對象的引用)是不可變的(const類型)。此函數(shù)經(jīng)常用在函數(shù)調(diào)用時用戶定義類型的值傳遞及返回。拷貝構(gòu)造函數(shù)要調(diào)用基類的拷貝構(gòu)造函數(shù)和成員函數(shù)。如果可以的話,它將用常量方式調(diào)用,另外,也可以用非常量方式調(diào)用。一般來說,調(diào)用拷貝構(gòu)造函數(shù)的情形 在C++中,下面三種對象需要調(diào)用拷貝構(gòu)造函數(shù):
2、 1)一個對象以值傳遞的方式傳入函數(shù)體; 2)一個對象以值傳遞的方式從函數(shù)返回; 3)一個對象需要通過另外一個對象進行初始化; 如果在前兩種情況不使用拷貝構(gòu)造函數(shù)的時候,就會導(dǎo)致一個指針指向已經(jīng)被刪除的內(nèi)存空間。對于第三種情況來說,初始化和賦值的不同含義是構(gòu)造函數(shù)調(diào)用的原因。事實上,拷貝構(gòu)造函數(shù)是由普通構(gòu)造函數(shù)和賦值操作符共同實現(xiàn)的。描述拷貝構(gòu)造函數(shù)和賦值運算符的異同的參考資料有很多?! 】截悩?gòu)造函數(shù)不可以改變它所引用的對象,其原因如下:當(dāng)一個對象以傳遞值的方式傳一個函數(shù)的時候,拷貝構(gòu)造函數(shù)自動的被調(diào)用來生成函數(shù)中的對象。如果一個對象是被傳入自己的拷貝構(gòu)造函數(shù),它
3、的拷貝構(gòu)造函數(shù)將會被調(diào)用來拷貝這個對象這樣復(fù)制才可以傳入它自己的拷貝構(gòu)造函數(shù),這會導(dǎo)致無限循環(huán)直至棧溢出(StackOverflow)。除了當(dāng)對象傳入函數(shù)的時候被隱式調(diào)用以外,拷貝構(gòu)造函數(shù)在對象被函數(shù)返回的時候也同樣的被調(diào)用。隱式的拷貝構(gòu)造函數(shù) 如果在類中沒有顯式的聲明一個拷貝構(gòu)造函數(shù),那么,編譯器會自動生成一個來進行對象之間的位拷貝(BitwiseCopy)。這個隱含的拷貝構(gòu)造函數(shù)簡單的關(guān)聯(lián)了所有的類成員。注意到這個隱式的拷貝構(gòu)造函數(shù)和顯式聲明的拷貝構(gòu)造函數(shù)的不同在于對成員的關(guān)聯(lián)方式。顯式聲明的拷貝構(gòu)造函數(shù)關(guān)聯(lián)的只是被實例化的類成員的缺省構(gòu)造函數(shù),除非另外一個構(gòu)造函
4、數(shù)在類初始化或構(gòu)造列表的時候被調(diào)用?! 】截悩?gòu)造函數(shù)使程序更有效率,因為它不用再構(gòu)造一個對象的時候改變構(gòu)造函數(shù)的參數(shù)列表。設(shè)計拷貝構(gòu)造函數(shù)是一個良好的風(fēng)格,即使是編譯系統(tǒng)會自動為你生成默認拷貝構(gòu)造函數(shù)。事實上,默認拷貝構(gòu)造函數(shù)可以應(yīng)付許多情況。示例 以下討論中將用到的例子: classCExample { public: CExample(){pBuffer=NULL;nSize=0;} ~CExample(){deletepBuffer;} voidInit(intn){pBuffer=newchar[n];nSize=n;} private: ch
5、ar*pBuffer;//類的對象中包含指針,指向動態(tài)分配的內(nèi)存資源 intnSize; }; 這個類的主要特點是包含指向其他資源的指針?! Buffer指向堆中分配的一段內(nèi)存空間?! ntmain(intargc,char*argv[]) { CExampletheObjone; theObjone.Init(40); //現(xiàn)在需要另一個對象,需要將他初始化成對象一的狀態(tài) CExampletheObjtwo=theObjone; ... } 語句"CExampletheObjtwo=theObjone;"用theObjone初始化theObj
6、two。 其完成方式是內(nèi)存拷貝,復(fù)制所有成員的值?! ⊥瓿珊?,theObjtwo.pBuffer==theObjone.pBuffer?! 〖此鼈儗⒅赶蛲瑯拥牡胤剑羔橂m然復(fù)制了,但所指向的空間并沒有復(fù)制,而是由兩個對象共用了。這樣不符合要求,對象之間不獨立了,并為空間的刪除帶來隱患。所以需要采用必要的手段來避免此類情況?! 』仡櫼幌麓苏Z句的具體過程:首先建立對象theObjtwo,并調(diào)用其構(gòu)造函數(shù),然后成員被拷貝?! 】梢栽跇?gòu)造函數(shù)中添加操作來解決指針成員的問題?! ∷訡++語法中除了提供缺省形式的構(gòu)造函數(shù)外,還規(guī)范了另一種特殊的構(gòu)造函數(shù):拷貝構(gòu)造函數(shù),上面的語句
7、中,如果類中定義了拷貝構(gòu)造函數(shù),這對象建立時,調(diào)用的將是拷貝構(gòu)造函數(shù),在拷貝構(gòu)造函數(shù)中,可以根據(jù)傳入的變量,復(fù)制指針所指向的資源。 拷貝構(gòu)造函數(shù)的格式為:構(gòu)造函數(shù)名(對象的引用) 提供了拷貝構(gòu)造函數(shù)后的CExample類定義為: classCExample { public: CExample(){pBuffer=NULL;nSize=0;} ~CExample(){deletepBuffer;} CExample(constCExample&);//拷貝構(gòu)造函數(shù) voidInit(intn){pBuffer=newch