C++為我們提供了默認(rèn)的復(fù)制構(gòu)造函數(shù),賦值函數(shù)和析構(gòu)函數(shù),使用的全部是“淺復(fù)制”,即僅僅復(fù)制棧上的數(shù)據(jù)。換句話(huà)說(shuō),如果我們涉及到了對(duì)堆數(shù)據(jù)的操作,這些函數(shù)都必須我們自己重新來(lái)寫(xiě)。我很郁悶為什么在編譯的時(shí)候,C++不能自己發(fā)現(xiàn)構(gòu)造函數(shù)使用了堆操作,從而提醒不要使用默認(rèn)的這三個(gè)函數(shù)。也許是因?yàn)橐幾g器做到智能的判斷很難吧。用new...delete或許很容易看出來(lái),但是更多的函數(shù)調(diào)用,特別是涉及到C風(fēng)格的函數(shù)的時(shí)候,真的很難判斷哪些函數(shù)使用到了堆操作。
而這三個(gè)函數(shù)的作用可以說(shuō)是巨大的!析構(gòu)就不說(shuō)了,析構(gòu)可以說(shuō)是C++永遠(yuǎn)的痛。復(fù)制構(gòu)造函數(shù)用得最多的地方,恐怕就是成員初始化列表的時(shí)候,這幾乎是在一個(gè)類(lèi)成員數(shù)據(jù)使用到另外一個(gè)類(lèi)對(duì)象時(shí)候的唯一方法。而賦值函數(shù)則是把數(shù)據(jù)從語(yǔ)句體(“{}”對(duì),循環(huán),判斷)中帶出的最簡(jiǎn)單方法——雖然我們現(xiàn)在可以很方便的使用vector。
這里先說(shuō)說(shuō)復(fù)制構(gòu)造函數(shù)吧。如果遺漏申明,又不慎用到,比如這個(gè)例子:
#include <iostream>
#include <vector>
class A
{
private:
int a;
public:
A(int _a): a(_a)
{
std::cout << "A created!\n";
}
/*
A(const A& copy): a(copy.a)
{
std::cout << "A copy created!\n";
}
*/
~A()
{
std::cout << "A destroyed!\n";
}
void show() const
{
std::cout << a << std::endl;
}
};
int main(int argc, char* argv[])
{
A a(1);
A b(a);
return 0;
}
那么,結(jié)果運(yùn)行就會(huì)出現(xiàn)貌似創(chuàng)建一次,但是卻銷(xiāo)毀了兩次的假象。這當(dāng)然是不可能發(fā)生的,但是郁悶的是,C++中的構(gòu)造和析構(gòu)不總是成對(duì)出現(xiàn)的,比如我們前面說(shuō)到的手動(dòng)顯式調(diào)用析構(gòu)函數(shù)的情況。所以,如果放在大的項(xiàng)目中,這為我們的調(diào)試帶來(lái)更多的困難。
所以,結(jié)論是,如果A類(lèi)構(gòu)造具有堆操作,有可能把A類(lèi)作為B類(lèi)的成員數(shù)據(jù),B類(lèi)又有可能通過(guò)成員初始化列表構(gòu)造A對(duì)象,請(qǐng)一定別忘記手寫(xiě)復(fù)制構(gòu)造函數(shù)。