您好,登錄后才能下訂單哦!
這篇文章主要為大家展示了“C++中深復制和淺復制是什么”,內容簡而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領大家一起研究并學習一下“C++中深復制和淺復制是什么”這篇文章吧。
在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
總之,上面的現象就是淺復制,這種淺復制很可能帶來不安全因素,這種不安全同樣體現在內存釋放時(同一個內存不能釋放兩次),所以需要使用下面將要介紹的深復制。
按照掌握的資料,深復制需要編寫賦值構造函數,創建成員指針所指向內存的新副本。比如上一節定義的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++中深復制和淺復制是什么”這篇文章的所有內容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內容對大家有所幫助,如果還想學習更多知識,歡迎關注億速云行業資訊頻道!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。