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