C++深復制和淺復制講解
前言:
深復制和淺復制在python中同樣存在,但是由於python中沒有指針這個概念,所以當時學python時理解這個問題非常困難。實際上,通過這些天的學習,越發的認為C++的語法雖然復雜,但是由於其每個概念都比較清晰反而比python更好理解。python中很多語法都有些“模糊”,尤其是它的魔法函數部分,使用的時候總是怕自己理解錯誤從而造成誤用。
1、什麼是淺復制
在C++中深復制和淺復制最大的區別在“類包含指針類型的數據成員”時。由於默認的復制構造函數完成的是對象成員的數值復制,當原對象含有指針P指向地址xxx時,通過原對象復制得到的新對象的指針P同樣指向地址xxx,造成同一個地址xxx被兩個對象同時指向,這是非常不安全的,因為任一對象對地址xxx的操作很可能會對另一個對象造成不良影響。下面定義的Duck類就包含指針成員foot,該成員指向數組的首地址:
class Duck { public: Duck() { foot = new int[2](); } int *foot; };
首先定義一個duck對象,然後通過duck
復制得到anotherDuck
,那麼二者的指針成員foot指向相同的地址。anotherDuck
對指針成員foot指向的內存進行操作,和duck的指針成員foot對指向的內存進行操作是等價的,因為兩個對象的foot指針都指向相同的地址,這就是不安全的來源:
Duck duck; Duck anotherDuck = duck; /// anotherDuck對foot指向內存進行賦值 anotherDuck.foot[1] = 666;
此外,為瞭證明原對象和新對象的foot指針都指向相同的地址,可以查看一下:
printf("their id is %p and %p \n", duck.foot, anotherDuck.foot); /// 得到的結果證明二者確實指向相同的地址 their id is 005B0CE8 and 005B0CE8
總之,上面的現象就是淺復制,這種淺復制很可能帶來不安全因素,這種不安全同樣體現在內存釋放時(同一個內存不能釋放兩次),所以需要使用下面將要介紹的深復制。
2、如何實現深復制
按照掌握的資料,深復制需要編寫賦值構造函數,創建成員指針所指向內存的新副本。比如上一節定義的Duck類指針成員foot,復制構造函數需要創建foot指向的內存的新副本:
Duck(Duck &duck) { /// 1、創建新的內存空間 foot = new int[2](); /// 2、將原對象的指針所指向的數組數值 /// 全部復制到新對象指針指向的數組 for (int i = 0; i < 2; i++) { foot[i] = duck.foot[i]; } }
其中第1步目的是創建新的內存空間,讓新對象的指針成員指向新的內存,而不是和原對象指向相同的內存,同時必須保證新的內存所存儲的類型和原對象相同,都是int類型的2個元素的數組。第2步的目的是將原對象指針所指向的數組的值,全部復制到新對象指所針指向的數組中。經過上面的兩個步驟,深度復制完成。
為瞭確保原對象duck和新對象anotherDuck的foot指針所指向的地址不同,可以進行下面的測試,輸出兩個foot指向的地址:
printf("their id is %p and %p \n", duck.foot, anotherDuck.foot); /// 結果顯示兩個foot指向的地址是不同的 their id is 01250FA0 and 01250B40
到此這篇關於C++深復制和淺復制講解的文章就介紹到這瞭,更多相關C++深復制和淺復制內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- C++類的繼承和派生及指針安全引用
- C++構造函數+復制構造函數+重載等號運算符調用
- C++函數指針+對象指針+this指針+指向類靜態和非靜態成員的指針
- C++二級指針和指針的引用
- C++標準庫封裝的vector數組