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