C++中的拷貝構(gòu)造詳解
拷貝構(gòu)造函數(shù)
同一個(gè)類的對象在內(nèi)存中有完全相同的結(jié)構(gòu),如果作為一個(gè)整體進(jìn)行復(fù)制或稱拷貝是完美可行的,這個(gè)拷貝過程只需要拷貝數(shù)據(jù)成員,而函數(shù)成員是公用的(只有一份代碼);在建立對象時(shí)可用同一類的另一個(gè)對象來初始化該對象的存儲空間,這時(shí)所用的構(gòu)造函數(shù)稱為拷貝構(gòu)造函數(shù)
例如:
class Object
{
int value;
public:
Object(int x = 0) :value(x) {}
~Object() {}
Object(Object& obj):value(obj.value)
{
cout << "Copy Create" << endl;
}
};
int main()
{
Object obja(10);
Object objb(obja);//一個(gè)對象初始化另一個(gè)對象空間,調(diào)用拷貝構(gòu)造
}
當(dāng)一個(gè)對象去初始化另一個(gè)對象空間,調(diào)用拷貝構(gòu)造;若類中沒有寫拷貝構(gòu)造,如同構(gòu)造函數(shù)與析構(gòu)函數(shù)一樣,系統(tǒng)會生成一個(gè)缺省的拷貝構(gòu)造函數(shù)
OBject(Object& obj)
{}
拷貝構(gòu)造中的引用
如果我們在寫的拷貝構(gòu)造不加引用,這樣會引起死遞歸
//Object(Object& obj):value(obj.value)
Object(Object obj):value(obj.value)
{
cout << "Copy Create" << endl;
}
為什么拷貝構(gòu)造函數(shù)必須采用引用傳參,否則會引發(fā)無窮遞歸呢?
這個(gè)問題其實(shí)很簡單,再復(fù)制對象時(shí)要分為兩個(gè)步驟:
第一步:開辟一個(gè)臨時(shí)空間;
第二步:由于臨時(shí)空間是需要構(gòu)造的,重新調(diào)用拷貝構(gòu)造函數(shù)(無窮遞歸形成…)
同時(shí)我們可以在拷貝構(gòu)造參數(shù)前加上一個(gè)引用,來限制可能會出現(xiàn)的問題
//Object(Object& obj):value(obj.value)
Object(const Object& obj):value(obj.value)
{
cout << "Copy Create" << endl;
}
//這里的const修飾,令我們不能修改被拷貝對象
什么情況會使用拷貝構(gòu)造
拷貝構(gòu)造不止在使用一個(gè)對象去構(gòu)造另一個(gè)對象時(shí)調(diào)用,在下面這些情況也會調(diào)用:
class Object{int value;public:Object(){cout << "Object::Object" << this << endl;}Object(int x = 0) :value(x){cout << "Object::Object" << this << endl;}~Object(){cout << "Objecet::~Object" << this << endl;}Object(Object& obj) :value(obj.value){cout << "Copy Create" << this << endl;}void SetValue(int x) { value = x; }int GetValue() const { return value; }};Object fun(Object obj){int val = obj.GetValue();Object obja(val);return obja;}int main(){Object objx(0);Object objy(0);objy = fun(objx);return 0;}class Object
{
int value;
public:
Object()
{
cout << "Object::Object" << this << endl;
}
Object(int x = 0) :value(x)
{
cout << "Object::Object" << this << endl;
}
~Object()
{
cout << "Objecet::~Object" << this << endl;
}
Object(Object& obj) :value(obj.value)
{
cout << "Copy Create" << this << endl;
}
void SetValue(int x) { value = x; }
int GetValue() const { return value; }
};
Object fun(Object obj)
{
int val = obj.GetValue();
Object obja(val);
return obja;
}
int main()
{
Object objx(0);
Object objy(0);
objy = fun(objx);
return 0;
}
在上面這一段代碼中,我們總共創(chuàng)建了幾個(gè)對象呢,我們來看一下

首先①②屬于對象的構(gòu)造,調(diào)用構(gòu)造函數(shù);程序運(yùn)行到objy = fun(objx);進(jìn)入到fun函數(shù),這是構(gòu)造臨時(shí)對象obj③屬于拷貝構(gòu)造;隨后④構(gòu)造對象obja;最后⑤這里也屬于拷貝構(gòu)造也需要?jiǎng)?chuàng)建一個(gè)臨時(shí)對象(將亡值)
并且我們無法將fun函數(shù)中obja對象之間return傳回給objy,因?yàn)樵诤瘮?shù)結(jié)束時(shí)obja會析構(gòu)失效,所以這里會創(chuàng)建一個(gè)新的臨時(shí)對象

總結(jié)
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
C語言中的運(yùn)算符優(yōu)先級和結(jié)合性一覽表
這篇文章主要介紹了C語言中的運(yùn)算符優(yōu)先級和結(jié)合性一覽表,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-02-02
C++實(shí)現(xiàn)leetcode(3.最長無重復(fù)字符的子串)
這篇文章主要介紹了C++實(shí)現(xiàn)leetcode(3.最長無重復(fù)字符的子串),本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07
C++實(shí)現(xiàn)LeetCode(151.翻轉(zhuǎn)字符串中的單詞)
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(151.翻轉(zhuǎn)字符串中的單詞),本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07
C++標(biāo)準(zhǔn)庫中sstream與strstream的區(qū)別詳細(xì)解析
以下是對C++標(biāo)準(zhǔn)庫中sstream與strstream的區(qū)別進(jìn)行了詳細(xì)的分析介紹,需要的朋友可以過來參考下2013-09-09
淺析string 與char* char[]之間的轉(zhuǎn)換
與char*不同的是,string不一定以NULL('\0')結(jié)束。string長度可以根據(jù)length()得到,string可以根據(jù)下標(biāo)訪問。所以,不能將string直接賦值給char*2013-10-10
Dijkstra算法最短路徑的C++實(shí)現(xiàn)與輸出路徑
今天小編就為大家分享一篇關(guān)于Dijkstra算法最短路徑的C++實(shí)現(xiàn)與輸出路徑,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧2019-02-02

