C++細(xì)講深淺拷貝與初始化列表如何操作
深淺拷貝區(qū)別
上一節(jié)簡(jiǎn)單提了編譯器會(huì)默認(rèn)給我們提供值拷貝構(gòu)造函數(shù),結(jié)果是新的對(duì)象會(huì)擁有和傳入對(duì)象一樣的屬性,由編譯器提供的拷貝構(gòu)造被稱(chēng)為淺拷貝構(gòu)造,而由我們自己編寫(xiě)的不同于編譯器提供的拷貝構(gòu)造函數(shù)就叫深拷貝構(gòu)造了,舉個(gè)典型的例子說(shuō)明。
代碼解釋
#include<iostream> using namespace std; //深淺拷貝問(wèn)題,存在經(jīng)典的坑,面試考過(guò) class Person { public: Person(int age,int height) { m_age = age; m_Height = new int(height); cout << "Person 的有參構(gòu)造函數(shù)調(diào)用" << endl; } //自己寫(xiě)不同于編譯器的拷貝構(gòu)造函數(shù)屬于深拷貝 Person(const Person& p) { cout << "拷貝構(gòu)造函數(shù)調(diào)用" << endl; m_age = p.m_age; //m_Height = p.m_Height; /*編譯器默認(rèn)執(zhí)行上行代碼 新開(kāi)辟的地址相同,會(huì)導(dǎo)致調(diào)用析構(gòu)函數(shù)時(shí)違法操作,無(wú)法訪問(wèn)內(nèi)存*/ m_Height =new int(*p.m_Height); } ~Person() { //析構(gòu)代碼,將堆區(qū)開(kāi)辟的數(shù)據(jù)做釋放操作 if (m_Height != NULL) { delete m_Height; m_Height = NULL; } cout << "~Person 的析構(gòu)構(gòu)造函數(shù)調(diào)用"<<endl; } int m_age; int* m_Height; }; void test() { Person p1(20,180); Person p2(p1); cout << "p2.age= " << p2.m_age << " p2.height=" << *p2.m_Height << endl; } int main() { test(); system("pause"); }
創(chuàng)建Person類(lèi),設(shè)置m_age和指針類(lèi)型*m_Height為私有屬性;依次對(duì)Person類(lèi)設(shè)置有參構(gòu)造和拷貝構(gòu)造函數(shù)以及析構(gòu)函數(shù);前面析構(gòu)函數(shù)一直沒(méi)有什么作用,其實(shí)它是用來(lái)清理對(duì)象的,析構(gòu)函數(shù)會(huì)在程序結(jié)束前自動(dòng)調(diào)用,這時(shí)候就可以使用delete清理掉;
特別注意
Person(const Person& p) { cout << "拷貝構(gòu)造函數(shù)調(diào)用" << endl; m_age = p.m_age; //m_Height = p.m_Height; /*編譯器默認(rèn)執(zhí)行上行代碼 新開(kāi)辟的地址相同,會(huì)導(dǎo)致調(diào)用析構(gòu)函數(shù)時(shí)違法操作,無(wú)法訪問(wèn)內(nèi)存*/ m_Height =new int(*p.m_Height); } ~Person() { //析構(gòu)代碼,將堆區(qū)開(kāi)辟的數(shù)據(jù)做釋放操作 if (m_Height != NULL) { delete m_Height; m_Height = NULL; } cout << "~Person 的析構(gòu)構(gòu)造函數(shù)調(diào)用"<<endl; }
這里不能使用編譯器提供的淺拷貝,如果直接使用m_Height=p.m_Height,毫無(wú)疑問(wèn)這兩個(gè)屬性地址相同,那么在調(diào)用析構(gòu)函數(shù)的時(shí)候,p1先釋放內(nèi)存,這時(shí)候雖然有一個(gè)NULL判斷,但是此塊內(nèi)存已經(jīng)被刪除,再次訪問(wèn)都會(huì)提示錯(cuò)誤,這是很危險(xiǎn)的,所以我們需要用深拷貝解決重復(fù)刪除的問(wèn)題。使用m_Height=new int(*p.m_Height) 語(yǔ)句給身高屬性重新開(kāi)辟空間,這樣在調(diào)用析構(gòu)的時(shí)候各自清理各自的屬性,就解決了這個(gè)淺拷貝帶來(lái)的重復(fù)清理問(wèn)題。
內(nèi)存圖解釋
上面是淺拷貝的p1、p2對(duì)象的內(nèi)存示意圖,兩次析構(gòu)會(huì)重復(fù)當(dāng)問(wèn)0x00011地址,但是當(dāng)這個(gè)地址被刪除后,是不允許再次訪問(wèn)的。
利用我們?cè)O(shè)置的深拷貝構(gòu)造后,地址不一樣,各自刪除各自的地址,解決問(wèn)題
初始化列表
初始化列表用來(lái)給屬性初始化
語(yǔ)法
普通構(gòu)造函數(shù)+:+ 類(lèi)屬性(變量或常量)+ {}
具體實(shí)現(xiàn)
class Person { public: Person() :m_age(20), m_sex(1), m_height(180){} Person(int a, int b, int c) :m_age(a), m_sex(b), m_height(c){} int m_age; int m_sex; int m_height; }; int main() { Person p1; Person p2(10, 20, 30); cout << "年齡為:" << p1.m_age; cout << "性別為:" << p1.m_sex; cout << "身高為:" << p1.m_height<<endl; cout << "年齡為:" << p2.m_age; cout << "性別為:" << p2.m_sex; cout << "身高為:" << p2.m_height << endl;; }
主函數(shù)中p1調(diào)用無(wú)參構(gòu)造函數(shù),各屬性初始化為屬性()括號(hào)里面的值;p2調(diào)用有參構(gòu)造函數(shù),將實(shí)參10,20,30分別傳給a,b,c,然后a的值傳給m_age,b的值傳給m_sex;c的值傳給m_height;直接來(lái)看結(jié)果:
總結(jié)
C++對(duì)象特性的深淺拷貝和初始化列表到這里就分享完了,那麼意味著對(duì)象的初始化和清理就結(jié)束了,到后面會(huì)講一下靜態(tài)成員和類(lèi)對(duì)象作為類(lèi)成員的一個(gè)案例,用于鞏固這部分知識(shí)點(diǎn),希望下篇文章也可以得到你們的青睞
到此這篇關(guān)于C++細(xì)講深淺拷貝與初始化列表如何操作的文章就介紹到這了,更多相關(guān)C++深淺拷貝內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++智能指針shared_ptr與weak_ptr的實(shí)現(xiàn)分析
shared_ptr是一個(gè)標(biāo)準(zhǔn)的共享所有權(quán)的智能指針,允許多個(gè)指針指向同一個(gè)對(duì)象,定義在 memory 文件中,命名空間為 std,這篇文章主要介紹了C++ 中 shared_ptr weak_ptr,需要的朋友可以參考下2022-09-09undefined reference to `SetPduPowerConsumptionCnt''錯(cuò)誤的解決方法
編譯時(shí)出現(xiàn)undefined reference to `SetPduPowerConsumptionCnt'錯(cuò)誤要如何解決呢?有沒(méi)有什么好的解決方法?下面小編就為大家解答吧,如果你也遇到了這種情況,可以過(guò)來(lái)參考下2013-07-07C++編程模板匹配超詳細(xì)的識(shí)別手寫(xiě)數(shù)字實(shí)現(xiàn)示例
大家好!本篇文章是關(guān)于手寫(xiě)數(shù)字識(shí)別的,接下來(lái)我將在這里記錄我的手寫(xiě)數(shù)字識(shí)別的從零到有,我在這里把我自己的寫(xiě)代碼過(guò)程發(fā)出來(lái),希望能幫到和我一樣努力求知的人2021-10-10C語(yǔ)言讀取文件流的相關(guān)函數(shù)用法簡(jiǎn)介
這篇文章主要介紹了C語(yǔ)言讀取文件流的相關(guān)函數(shù)用法簡(jiǎn)介,包括fread()函數(shù)和feof()函數(shù)的使用,需要的朋友可以參考下2015-08-08C++基于棧的深搜算法實(shí)現(xiàn)馬踏棋盤(pán)
這篇文章主要為大家詳細(xì)介紹了C++基于棧的深搜算法實(shí)現(xiàn)馬踏棋盤(pán),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-02-02C++實(shí)現(xiàn)LeetCode(17.電話號(hào)碼的字母組合)
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(17.電話號(hào)碼的字母組合),本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07