日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學(xué)無(wú)先后,達(dá)者為師

網(wǎng)站首頁(yè) 編程語(yǔ)言 正文

C++細(xì)講深淺拷貝與初始化列表如何操作_C 語(yǔ)言

作者:對(duì)象new不出來(lái) ? 更新時(shí)間: 2022-07-25 編程語(yǔ)言

深淺拷貝區(qū)別

上一節(jié)簡(jiǎn)單提了編譯器會(huì)默認(rèn)給我們提供值拷貝構(gòu)造函數(shù),結(jié)果是新的對(duì)象會(huì)擁有和傳入對(duì)象一樣的屬性,由編譯器提供的拷貝構(gòu)造被稱為淺拷貝構(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類,設(shè)置m_age和指針類型*m_Height為私有屬性;依次對(duì)Person類設(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ù)+:+ 類屬性(變量或常量)+ {}

具體實(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)成員和類對(duì)象作為類成員的一個(gè)案例,用于鞏固這部分知識(shí)點(diǎn),希望下篇文章也可以得到你們的青睞

原文鏈接:https://blog.csdn.net/m0_58618795/article/details/124912579

欄目分類
最近更新