溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

C++中怎么實現(xiàn)淺拷貝

發(fā)布時間:2021-07-19 17:44:04 來源:億速云 閱讀:216 作者:Leah 欄目:編程語言

這篇文章給大家介紹C++中怎么實現(xiàn)淺拷貝,內(nèi)容非常詳細,感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助。

C++淺拷貝就是成員數(shù)據(jù)之間的一一賦值:把值賦給一一賦給要拷貝的值。但是可能會有這樣的情況:對象還包含資源,這里的資源可以值堆資源,或者一個文件。。當值拷貝的時候,兩個對象就有用共同的資源,同時對資源可以訪問,這樣就會出問題。深拷貝就是用來解決這樣的問題的,它把資源也賦值一次,使對象擁有不同的資源,但資源的內(nèi)容是一樣的。對于堆資源來說,就是在開辟一片堆內(nèi)存,把原來的內(nèi)容拷貝。

如果你拷貝的對象中引用了某個外部的內(nèi)容(比如分配在堆上的數(shù)據(jù)),那么在拷貝這個對象的時候,讓新舊兩個對象指向同一個外部的內(nèi)容,就是C++淺拷貝;如果在拷貝這個對象的時候為新對象制作了外部對象的獨立拷貝,就是深拷貝

引用和指針的語義是相似的,引用是不可改變的指針,指針是可以改變的引用。其實都是實現(xiàn)了引用語義。 深拷貝和淺拷貝的區(qū)別是在對象狀態(tài)中包含其它對象的引用的時候,當拷貝一個對象時,如果需要拷貝這個對象引用的對象,則是深拷貝,否則是淺拷貝。

COW語義是“深拷貝”與“推遲計算”的組合,仍然是深拷貝,而非淺拷貝,因為拷貝之后的兩個對象的數(shù)據(jù)在邏輯上是不相關(guān)的,只是內(nèi)容相同。

無論深淺,都是需要的。當深拷貝發(fā)生時,通常表明存在著一個“聚合關(guān)系”,而C++淺拷貝發(fā)生時,通常表明存在著一個“相識關(guān)系”。
舉個簡單的例子:

當你實現(xiàn)一個Composite Pattern,你通常都會實現(xiàn)一個深拷貝(如果需要拷貝的話),很少有要求同的Composite共享Leaf的;

而當你實現(xiàn)一個Observer Pattern時,如果你需要拷貝Observer,你大概不會去拷貝Subject,這時就要實現(xiàn)個C++淺拷貝。 是深拷貝還是淺拷貝,并不是取決于時間效率、空間效率或是語言等等,而是取決于哪一個是邏輯上正確的

在學(xué)習(xí)這一章內(nèi)容前我們已經(jīng)學(xué)習(xí)過了類的構(gòu)造函數(shù)和析構(gòu)函數(shù)的相關(guān)知識,對于普通類型的對象來說,他們之間的復(fù)制是很簡單的,例如:

int a = 10; int b =a;

在C++淺拷貝中,自己定義的類的對象同樣是對象,誰也不能阻止我們用以下的方式進行復(fù)制,例如:

#include <iostream> usingnamespacestd;   classTest   {   public: Test(inttemp)   {   p1=temp;   }  protected: intp1;   };   voidmain()   {   Test a(99);   Test b=a;   }

普通對象和類對象同為對象,他們之間的特性有相似之處也有不同之處,類對象內(nèi)部存在成員變量,而普通對象是沒有的,當同樣的復(fù)制方法發(fā)生在不同的對象上的時候,那么系統(tǒng)對他們進行的操作也是不一樣的,就類對象而言,相同類型的類對象是通過拷貝構(gòu)造函數(shù)來完成整個復(fù)制過程的,在上面的代碼中,我們并沒有看到拷貝構(gòu)造函數(shù),同樣完成了復(fù)制工作,這又是為什么呢?因為當一個類沒有自定義的拷貝構(gòu)造函數(shù)的時候系統(tǒng)會自動提供一個默認的拷貝構(gòu)造函數(shù),來完成復(fù)制工作。

下面,我們?yōu)榱苏f明情況,就普通情況而言(以上面的代碼為例),我們來自己定義一個與系統(tǒng)默認拷貝構(gòu)造函數(shù)一樣的拷貝構(gòu)造函數(shù),看看它的內(nèi)部是如何工作的!

代碼如下:

#include <iostream> usingnamespacestd;   classTest {   public: Test(inttemp)   { p1=temp; }   Test(Test &c_t)  //這里就是自定義的拷貝構(gòu)造函數(shù)   {   cout<<"進入copy構(gòu)造函數(shù)"<p1p1=c_t.p1;  //這句如果去掉就不能完成復(fù)制工作了,此句復(fù)制過程的核心語句 } public: intp1; };   voidmain() { Test a(99); Test b=a;   cout<cin.get();   }

上面代碼中的Test(Test &c_t)就是我們自定義的拷貝構(gòu)造函數(shù),拷貝構(gòu)造函數(shù)的名稱必須與類名稱一致,函數(shù)的形式參數(shù)是本類型的一個引用變量,且必須是引用。

當用一個已經(jīng)初始化過了的自定義類類型對象去初始化另一個新構(gòu)造的對象的時候,拷貝構(gòu)造函數(shù)就會被自動調(diào)用,如果你沒有自定義拷貝構(gòu)造函數(shù)的時候系統(tǒng)將會提供給一個默認的拷貝構(gòu)造函數(shù)來完成這個過程,上面代碼的復(fù)制核心語句就是通過 Test(Test &c_t)拷貝構(gòu)造函數(shù)內(nèi)的p1=c_t.p1;語句完成的。如果取掉這句代碼,那么b對象的p1屬性將得到一個未知的隨機值。

關(guān)于C++中怎么實現(xiàn)淺拷貝就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。

向AI問一下細節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

c++
AI