C++深復(fù)制和淺復(fù)制講解
前言:
深復(fù)制和淺復(fù)制在python中同樣存在,但是由于python中沒有指針這個概念,所以當(dāng)時學(xué)python時理解這個問題非常困難。實(shí)際上,通過這些天的學(xué)習(xí),越發(fā)的認(rèn)為C++的語法雖然復(fù)雜,但是由于其每個概念都比較清晰反而比python更好理解。python中很多語法都有些“模糊”,尤其是它的魔法函數(shù)部分,使用的時候總是怕自己理解錯誤從而造成誤用。
1、什么是淺復(fù)制
在C++中深復(fù)制和淺復(fù)制最大的區(qū)別在“類包含指針類型的數(shù)據(jù)成員”時。由于默認(rèn)的復(fù)制構(gòu)造函數(shù)完成的是對象成員的數(shù)值復(fù)制,當(dāng)原對象含有指針P指向地址xxx時,通過原對象復(fù)制得到的新對象的指針P同樣指向地址xxx,造成同一個地址xxx被兩個對象同時指向,這是非常不安全的,因?yàn)槿我粚ο髮Φ刂穢xx的操作很可能會對另一個對象造成不良影響。下面定義的Duck類就包含指針成員foot,該成員指向數(shù)組的首地址:
class Duck { public: ?? ?Duck() { foot = new int[2](); } ?? ?int *foot; };
首先定義一個duck對象,然后通過duck
復(fù)制得到anotherDuck
,那么二者的指針成員foot指向相同的地址。anotherDuck
對指針成員foot指向的內(nèi)存進(jìn)行操作,和duck的指針成員foot對指向的內(nèi)存進(jìn)行操作是等價的,因?yàn)閮蓚€對象的foot指針都指向相同的地址,這就是不安全的來源:
Duck duck; Duck anotherDuck = duck; /// anotherDuck對foot指向內(nèi)存進(jìn)行賦值 anotherDuck.foot[1] = 666;
此外,為了證明原對象和新對象的foot指針都指向相同的地址,可以查看一下:
printf("their id is %p and %p \n", duck.foot, anotherDuck.foot); /// 得到的結(jié)果證明二者確實(shí)指向相同的地址 their id is 005B0CE8 and 005B0CE8
總之,上面的現(xiàn)象就是淺復(fù)制,這種淺復(fù)制很可能帶來不安全因素,這種不安全同樣體現(xiàn)在內(nèi)存釋放時(同一個內(nèi)存不能釋放兩次),所以需要使用下面將要介紹的深復(fù)制。
2、如何實(shí)現(xiàn)深復(fù)制
按照掌握的資料,深復(fù)制需要編寫賦值構(gòu)造函數(shù),創(chuàng)建成員指針?biāo)赶騼?nèi)存的新副本。比如上一節(jié)定義的Duck類指針成員foot,復(fù)制構(gòu)造函數(shù)需要創(chuàng)建foot指向的內(nèi)存的新副本:
Duck(Duck &duck) { ? ? /// 1、創(chuàng)建新的內(nèi)存空間 ? ? foot = new int[2](); ? ? /// 2、將原對象的指針?biāo)赶虻臄?shù)組數(shù)值 ? ? /// 全部復(fù)制到新對象指針指向的數(shù)組 ? ? for (int i = 0; i < 2; i++) { ? ? ? ? foot[i] = duck.foot[i]; ? ? } }
其中第1步目的是創(chuàng)建新的內(nèi)存空間,讓新對象的指針成員指向新的內(nèi)存,而不是和原對象指向相同的內(nèi)存,同時必須保證新的內(nèi)存所存儲的類型和原對象相同,都是int類型的2個元素的數(shù)組。第2步的目的是將原對象指針?biāo)赶虻臄?shù)組的值,全部復(fù)制到新對象指所針指向的數(shù)組中。經(jīng)過上面的兩個步驟,深度復(fù)制完成。
為了確保原對象duck和新對象anotherDuck的foot指針?biāo)赶虻牡刂凡煌梢赃M(jìn)行下面的測試,輸出兩個foot指向的地址:
printf("their id is %p and %p \n", duck.foot, anotherDuck.foot); /// 結(jié)果顯示兩個foot指向的地址是不同的 their id is 01250FA0 and 01250B40
到此這篇關(guān)于C++深復(fù)制和淺復(fù)制講解的文章就介紹到這了,更多相關(guān)C++深復(fù)制和淺復(fù)制內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++數(shù)據(jù)結(jié)構(gòu)之AVL樹的實(shí)現(xiàn)
AVL樹是高度平衡的而二叉樹,它的特點(diǎn)是AVL樹中任何節(jié)點(diǎn)的兩個子樹的高度最大差別為1,本文主要給大家介紹了C++如何實(shí)現(xiàn)AVL樹,需要的朋友可以參考下2022-06-06C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
這篇文章主要介紹了C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法的相關(guān)資料,希望通過本文能幫助到大家,需要的朋友可以參考下2017-09-09c++連接mysql數(shù)據(jù)庫的兩種方法(ADO連接和mysql api連接)
現(xiàn)在正做一個接口,通過不同的連接字符串操作不同的數(shù)據(jù)庫。要用到mysql數(shù)據(jù)庫,C++連接mysql有2種方法:利用ADO連接、利用mysql自己的api函數(shù)進(jìn)行連接,下面看看如何用吧2013-12-12全面解析C++中的new,operator new與placement new
以下是C++中的new,operator new與placement new進(jìn)行了詳細(xì)的說明介紹,需要的朋友可以過來參考下2013-09-09函數(shù)指針的強(qiáng)制類型轉(zhuǎn)換實(shí)現(xiàn)代碼
函數(shù)指針的強(qiáng)制類型轉(zhuǎn)換實(shí)現(xiàn)代碼。需要的朋友可以過來參考下,希望對大家有所幫助2013-10-10C語言素?cái)?shù)(質(zhì)數(shù))判斷的3種方法舉例
這篇文章主要給大家介紹了關(guān)于C語言素?cái)?shù)(質(zhì)數(shù))判斷的3種方法,質(zhì)數(shù)是只能被1或者自身整除的自然數(shù)(不包括1),稱為質(zhì)數(shù),文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-11-11c++中的volatile和variant關(guān)鍵字詳解
大家好,本篇文章主要講的是c++中的volatile和variant關(guān)鍵字詳解,感興趣的同學(xué)趕快來看一看吧,對你有幫助的話記得收藏一下2022-01-01