欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++超詳細(xì)講解拷貝構(gòu)造函數(shù)

 更新時(shí)間:2022年06月02日 08:54:33   作者:iheal  
我們經(jīng)常會(huì)用一個(gè)變量去初始化一個(gè)同類(lèi)型的變量,那么對(duì)于自定義的類(lèi)型也應(yīng)該有類(lèi)似的操作,那么創(chuàng)建對(duì)象時(shí)如何使用一個(gè)已經(jīng)存在的對(duì)象去創(chuàng)建另一個(gè)與之相同的對(duì)象呢

構(gòu)造函數(shù)

只有單個(gè)形參,該形參是對(duì)本類(lèi)類(lèi)型對(duì)象的引用(一般常用const修飾),在用已存在的類(lèi)類(lèi)型對(duì)象創(chuàng)建新對(duì)象時(shí)由編譯器自動(dòng)調(diào)用

拷貝構(gòu)造函數(shù)是構(gòu)造函數(shù)的一個(gè)重載,因此顯式的定義了拷貝構(gòu)造,那么編譯器也不再默認(rèn)生成構(gòu)造函數(shù)。

特征

拷貝構(gòu)造也是一個(gè)特殊的成員函數(shù)

特征如下:

  • 拷貝構(gòu)造是構(gòu)造函數(shù)的一個(gè)重載;
  • 拷貝構(gòu)造的參數(shù)只有一個(gè)并且類(lèi)型必須是該類(lèi)的引用,而不是使用傳值調(diào)用,否則會(huì)無(wú)限遞歸;
  • 若沒(méi)有顯式定義拷貝構(gòu)造函數(shù),編譯器會(huì)自己生成一個(gè)默認(rèn)拷貝構(gòu)造,默認(rèn)的拷貝構(gòu)造函數(shù)對(duì)象按按內(nèi)存存儲(chǔ)和字節(jié)序完成拷貝,也叫淺拷貝;
class Date
{
public:
	Date(int year, int month, int day)
		:
		_year(year),
		_month(month),
		_day(day)
	{}
	void Display()
	{
		cout << _year << "-" << _month << "-" << _day << endl;
	}
private:
	int _year;
	int _month;
	int _day;
};
int main()
{
	Date d1(2001, 7, 28);
	Date d2(d1);
	d1.Display();
	d2.Display();
	return 0;
}

輸出:

2001-7-28

2001-7-28

對(duì)于那些直接管理著內(nèi)存資源的類(lèi)(含有指針變量),那么簡(jiǎn)單的值拷貝還頂?shù)米??顯然頂不住啊。

通過(guò)圖示說(shuō)明:

兩個(gè)string類(lèi)的對(duì)象指向了同一塊空間,這不就亂套了嗎,如果其中一個(gè)對(duì)象通過(guò)指針改變了指向內(nèi)存的數(shù)據(jù),那么另一個(gè)對(duì)象也會(huì)受到影響,這是我們不愿發(fā)生的,我們希望每個(gè)對(duì)象都能獨(dú)立運(yùn)作。

下面這個(gè)程序會(huì)崩潰

class String
{
public:
	String(const char* str = "songxin")
	{
		cout << "String(const char* str = \"songxin\")" << endl;
		_str = (char*)malloc(strlen(str) + 1);
		strcpy(_str, str);
	}
	~String()
	{
		cout << "~String()" << endl;
		free(_str);
		_str = nullptr;
	}
private:
	char* _str;
};
int main()
{
	String s1;
	String s2(s1);
	return 0;
}

原因是兩個(gè)string類(lèi)的成員指針都指向一塊內(nèi)存,而它們又分別調(diào)用了一次析構(gòu)函數(shù),相當(dāng)于對(duì)同一塊內(nèi)存空間釋放了兩次,程序崩潰。

因此對(duì)于這種情況的對(duì)象,我們就不能再使用編譯器生成的默認(rèn)拷貝構(gòu)造了,而只能自己去顯式的定義拷貝構(gòu)造并且要實(shí)現(xiàn)深拷貝。

編譯器生成的拷貝構(gòu)造

編譯器默認(rèn)生成的拷貝構(gòu)造會(huì)做些什么呢?

  • 對(duì)于內(nèi)置類(lèi)型成員

? 完成值拷貝;

  • 對(duì)于自定義類(lèi)型成員

調(diào)用成員的拷貝構(gòu)造;

class Time
{
public:
	Time(int hour = 0, int minute = 0, int second = 0)
		:
		_hour(hour),
		_minute(minute),
		_second(second)
	{}
	Time(Time& t)
	{
		_hour = t._hour;
		_minute = t._minute;
		_second = t._second;
	}
private:
	int _hour;
	int _minute;
	int _second;
};
Time top(0, 1, 1);
class Date
{
public:
	Date(int year = 1900, int month = 1, int day = 1, Time& t = top)
		:
		_year(year),
		_month(month),
		_day(day),
		_t(t)
	{}
private:
	int _year;
	int _month;
	int _day;
	Time _t;
};
int main()
{
	Time t(1, 1, 1);
	Date d1(2001, 7, 28,t);
	Date d2(d1);
	return 0;
}

如果默認(rèn)生成的拷貝構(gòu)造沒(méi)有調(diào)用Time類(lèi)成員的拷貝構(gòu)造,那么d2的_t的值應(yīng)該是(_hour = 0, _minute = 0, _second = 0),而這里最終的結(jié)果是d2中的_t和d2中的_t值相同。

這里Date類(lèi)中自動(dòng)生成的拷貝構(gòu)造函數(shù)的內(nèi)置類(lèi)型會(huì)進(jìn)行字節(jié)序拷貝,而對(duì)于自定義類(lèi)型_t調(diào)用了Time的拷貝構(gòu)造函數(shù)。

拷貝構(gòu)造的初始化列表

拷貝構(gòu)造是構(gòu)造函數(shù)的一個(gè)重載,因此拷貝構(gòu)造函數(shù)也是有初始化列表的,所以也建議在初始化列表階段完成對(duì)對(duì)象的初始化,養(yǎng)成良好習(xí)慣。

可以不顯式定義拷貝構(gòu)造函數(shù)的情況

  • 成員變量沒(méi)有指針;
  • 成員有指針,但并沒(méi)有管理內(nèi)存資源;

顯式定義拷貝構(gòu)造的誤區(qū)

之前一直存在這個(gè)誤區(qū):

我們都知道,編譯器生成的構(gòu)造函數(shù)在初始化列表會(huì)調(diào)用成員的構(gòu)造函數(shù),而我們顯式去定義構(gòu)造函數(shù)時(shí),即使我們不寫(xiě)也會(huì)在初始化列表去調(diào)用自定義類(lèi)型成員的構(gòu)造函數(shù)。

通過(guò)類(lèi)比,我就犯了一個(gè)低級(jí)錯(cuò)誤:

就是既然編譯器生成的拷貝構(gòu)造可以在初始化列表自動(dòng)調(diào)用自定義成員的拷貝構(gòu)造,那么我們顯式定義的拷貝構(gòu)造即使不寫(xiě),也會(huì)在初始化列表自動(dòng)去調(diào)用自定義成員的拷貝構(gòu)造。

于是我寫(xiě)出了如下代碼:

class Time
{
public:
	Time(int hour = 0, int minute = 0, int second = 0)
		:
		_hour(hour),
		_minute(minute),
		_second(second)
	{}
	Time(Time& t)
	{
		_hour = t._hour;
		_minute = t._minute;
		_second = t._second;
	}
private:
	int _hour;
	int _minute;
	int _second;
};
Time top(2, 2, 2);
class Date
{
public:
	Date(int year = 1900, int month = 1, int day = 1, Time& t = top)
		:
		_year(year),
		_month(month),
		_day(day),
		_t(t)
	{}
	Date(Date& d)//顯式定義了拷貝構(gòu)造
	{
	}
private:
	int _year;
	int _month;
	int _day;
	Time _t;
};
int main()
{
	Time t(1, 1, 1);
	Date d1(2001, 7, 28,t);
	Date d2(d1);
	return 0;
}

通過(guò)監(jiān)視窗口查看d2調(diào)用拷貝構(gòu)造后的值:

并沒(méi)有拷貝成功。

我只顧著類(lèi)比它們的功能,可我恰恰忽略了拷貝構(gòu)造也是一種構(gòu)造函數(shù)啊,那么自然的初始化列表也是和普通構(gòu)造一樣,會(huì)去調(diào)用自定義類(lèi)的構(gòu)造函數(shù),不處理內(nèi)置類(lèi)型。只不過(guò)編譯器生成的是經(jīng)過(guò)處理的構(gòu)造函數(shù)達(dá)到了拷貝的效果。(太傻逼了這錯(cuò)誤)

結(jié)論

拷貝構(gòu)造函數(shù)是構(gòu)造函數(shù)的一種,它也有初始化列表,如果是編譯器生成的拷貝構(gòu)造,它會(huì)對(duì)內(nèi)置類(lèi)型做字節(jié)序拷貝,對(duì)自定義類(lèi)型成員會(huì)調(diào)用自定義成員的拷貝構(gòu)造。

可如果是我們顯式定義出的拷貝構(gòu)造,它也是有初始化列表的,但是它的初始化列表可不會(huì)去調(diào)用成員的拷貝構(gòu)造奧,而是和普通構(gòu)造函數(shù)一樣,對(duì)于內(nèi)置類(lèi)型成員不去初始化值,對(duì)于自定義類(lèi)型成員調(diào)用自定義成員的構(gòu)造函數(shù)而不是拷貝構(gòu)造函數(shù)。

到此這篇關(guān)于C++超詳細(xì)講解拷貝構(gòu)造函數(shù)的文章就介紹到這了,更多相關(guān)C++拷貝構(gòu)造函數(shù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C語(yǔ)言中的getchar和putchar的使用方法

    C語(yǔ)言中的getchar和putchar的使用方法

    這篇文章主要介紹了C語(yǔ)言中的getchar和putchar的使用方法的相關(guān)資料,希望通過(guò)本文能幫助到大家,需要的朋友可以參考下
    2017-10-10
  • C語(yǔ)言實(shí)現(xiàn)快速排序

    C語(yǔ)言實(shí)現(xiàn)快速排序

    這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)快速排序算法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-08-08
  • C 語(yǔ)言指針概念的詳解

    C 語(yǔ)言指針概念的詳解

    這里主要介紹C 語(yǔ)言指針,這里整理了詳細(xì)的資料,對(duì)指針做了詳細(xì)說(shuō)明及簡(jiǎn)單示例代碼幫助大家理解什么是指針,有興趣的小伙伴可以參考下
    2016-08-08
  • c語(yǔ)言冒泡排序和選擇排序的使用代碼

    c語(yǔ)言冒泡排序和選擇排序的使用代碼

    算法中排序是十分重要的,而每一個(gè)學(xué)習(xí)計(jì)算機(jī)的都會(huì)在初期的時(shí)候接觸到這種排序,下面這篇文章主要給大家介紹了關(guān)于c語(yǔ)言冒泡排序和選擇排序使用的相關(guān)資料,需要的朋友可以參考下
    2022-04-04
  • C++實(shí)現(xiàn)圖書(shū)管理系統(tǒng)源碼

    C++實(shí)現(xiàn)圖書(shū)管理系統(tǒng)源碼

    這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)圖書(shū)管理系統(tǒng)源碼,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-03-03
  • C++實(shí)現(xiàn)簡(jiǎn)單通訊錄管理系統(tǒng)

    C++實(shí)現(xiàn)簡(jiǎn)單通訊錄管理系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)簡(jiǎn)單通訊錄管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-02-02
  • C++中String類(lèi)型的逆序方式

    C++中String類(lèi)型的逆序方式

    這篇文章主要介紹了C++中String類(lèi)型的逆序方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-07-07
  • C語(yǔ)言數(shù)據(jù)的存儲(chǔ)和取出詳細(xì)講解

    C語(yǔ)言數(shù)據(jù)的存儲(chǔ)和取出詳細(xì)講解

    這篇文章主要介紹了C語(yǔ)言數(shù)據(jù)的存儲(chǔ)和取出詳細(xì)講解,作者使用圖文代碼實(shí)例講解,有感興趣的同學(xué)可以學(xué)習(xí)研究下
    2021-02-02
  • C++核心編程之占位參數(shù)和默認(rèn)參數(shù)

    C++核心編程之占位參數(shù)和默認(rèn)參數(shù)

    這篇文章主要介紹了C++核心編程之占位參數(shù)和默認(rèn)參數(shù),c++中函數(shù)的形參列表中的形參是可以有默認(rèn)值的,函數(shù)的形參列表里可以有占位參數(shù),用來(lái)占位,調(diào)用函數(shù)時(shí)必須填補(bǔ)位置。下面更多相關(guān)內(nèi)容的詳細(xì)介紹,需要的小伙伴可以參考一下
    2022-03-03
  • C語(yǔ)言實(shí)現(xiàn)控制臺(tái)版貪吃蛇游戲

    C語(yǔ)言實(shí)現(xiàn)控制臺(tái)版貪吃蛇游戲

    這篇文章主要為大家詳細(xì)介紹了c語(yǔ)言貪吃蛇控制臺(tái)版,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-07-07

最新評(píng)論