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

學無先后,達者為師

網站首頁 編程語言 正文

C++細講深淺拷貝與初始化列表如何操作_C 語言

作者:對象new不出來 ? 更新時間: 2022-07-25 編程語言

深淺拷貝區別

上一節簡單提了編譯器會默認給我們提供值拷貝構造函數,結果是新的對象會擁有和傳入對象一樣的屬性,由編譯器提供的拷貝構造被稱為淺拷貝構造,而由我們自己編寫的不同于編譯器提供的拷貝構造函數就叫深拷貝構造了,舉個典型的例子說明。

代碼解釋

#include<iostream>
using namespace std;
//深淺拷貝問題,存在經典的坑,面試考過
class Person
{
public:
	Person(int age,int height)
	{
		m_age = age;
		m_Height = new int(height);
		cout << "Person 的有參構造函數調用" << endl;
	}
	//自己寫不同于編譯器的拷貝構造函數屬于深拷貝
	Person(const Person& p)
	{
		cout << "拷貝構造函數調用" << endl;
		m_age = p.m_age;
		//m_Height = p.m_Height;
		/*編譯器默認執行上行代碼
		新開辟的地址相同,會導致調用析構函數時違法操作,無法訪問內存*/
		m_Height =new int(*p.m_Height);
	}
	~Person()
	{
		//析構代碼,將堆區開辟的數據做釋放操作
		if (m_Height != NULL)
		{
			delete m_Height;
			m_Height = NULL;
		}
		cout << "~Person 的析構構造函數調用"<<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");
}

創建Person類,設置m_age和指針類型*m_Height為私有屬性;依次對Person類設置有參構造和拷貝構造函數以及析構函數;前面析構函數一直沒有什么作用,其實它是用來清理對象的,析構函數會在程序結束前自動調用,這時候就可以使用delete清理掉;

特別注意

Person(const Person& p)
	{
		cout << "拷貝構造函數調用" << endl;
		m_age = p.m_age;
		//m_Height = p.m_Height;
		/*編譯器默認執行上行代碼
		新開辟的地址相同,會導致調用析構函數時違法操作,無法訪問內存*/
		m_Height =new int(*p.m_Height);
	}
	~Person()
	{
		//析構代碼,將堆區開辟的數據做釋放操作
		if (m_Height != NULL)
		{
			delete m_Height;
			m_Height = NULL;
		}
		cout << "~Person 的析構構造函數調用"<<endl;
	}

這里不能使用編譯器提供的淺拷貝,如果直接使用m_Height=p.m_Height,毫無疑問這兩個屬性地址相同,那么在調用析構函數的時候,p1先釋放內存,這時候雖然有一個NULL判斷,但是此塊內存已經被刪除,再次訪問都會提示錯誤,這是很危險的,所以我們需要用深拷貝解決重復刪除的問題。使用m_Height=new int(*p.m_Height) 語句給身高屬性重新開辟空間,這樣在調用析構的時候各自清理各自的屬性,就解決了這個淺拷貝帶來的重復清理問題。

內存圖解釋

上面是淺拷貝的p1、p2對象的內存示意圖,兩次析構會重復當問0x00011地址,但是當這個地址被刪除后,是不允許再次訪問的。

利用我們設置的深拷貝構造后,地址不一樣,各自刪除各自的地址,解決問題

初始化列表

初始化列表用來給屬性初始化

語法

普通構造函數+:+ 類屬性(變量或常量)+ {}

具體實現

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;;
}

主函數中p1調用無參構造函數,各屬性初始化為屬性()括號里面的值;p2調用有參構造函數,將實參10,20,30分別傳給a,b,c,然后a的值傳給m_age,b的值傳給m_sex;c的值傳給m_height;直接來看結果:

總結

C++對象特性的深淺拷貝和初始化列表到這里就分享完了,那麼意味著對象的初始化和清理就結束了,到后面會講一下靜態成員和類對象作為類成員的一個案例,用于鞏固這部分知識點,希望下篇文章也可以得到你們的青睞

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

欄目分類
最近更新