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