拷貝建構函式,深拷貝,淺拷貝
深淺拷貝的區別:
淺拷貝是將原始物件中的資料型欄位拷貝到新物件中去,將引用型欄位的“引用”複製到新物件中去,不把“引用的物件”複製進去,所以原始物件和新物件引用同一物件,新物件中的引用型欄位發生變化會導致原始物件中的對應欄位也發生變化。
程式碼:
Line::Line(const Line &obj) {
cout << "呼叫拷貝建構函式併為指標 ptr 分配記憶體" << endl;
ptr = new int; *ptr = *obj.ptr; // 拷貝值,存在指標 }
深拷貝是在引用方面不同,深拷貝就是建立一個新的和原始欄位的內容相同的欄位,是兩個一樣大的資料段,所以兩者的引用是不同的,之後的新物件中的引用型欄位發生改變,
程式碼:CA(const CA& C)
{
a=C.a;
str=new char[a]; //深拷貝,沒有指標
if(str!=0)
strcpy(str,C.str);
}
拷貝建構函式是一種特殊的建構函式,它在建立物件時,是使用同一類中之前建立的物件來初始化新建立的物件。拷貝建構函式通常用於:
-
通過使用另一個同類型的物件來初始化新建立的物件。
-
複製物件把它作為引數傳遞給函式。
-
複製物件,並從函式返回這個物件。
如果在類中沒有定義拷貝建構函式,編譯器會自行定義一個。如果類帶有指標變數,並有動態記憶體分配,則它必須有一個拷貝建構函式。拷貝建構函式的最常見形式如下:
classname (const classname &obj) { // 建構函式的主體 }
在這裡,obj 是一個物件引用,該物件是用於初始化另一個物件的。
例項
#include <iostream> using namespace std; class Line { public: int getLength( void ); Line( int len ); // 簡單的建構函式 Line( const Line &obj); // 拷貝建構函式 ~Line(); // 解構函式 private: int *ptr; }; // 成員函式定義,包括建構函式 Line::Line(int len) { cout << "呼叫建構函式" << endl; // 為指標分配記憶體 ptr = new int; *ptr = len; } Line::Line(const Line &obj) { cout << "呼叫拷貝建構函式併為指標 ptr 分配記憶體" << endl; ptr = new int; *ptr = *obj.ptr; // 拷貝值 } Line::~Line(void) { cout << "釋放記憶體" << endl; delete ptr; } int Line::getLength( void ) { return *ptr; } void display(Line obj) { cout << "line 大小 : " << obj.getLength() <<endl; } // 程式的主函式 int main( ) { Line line(10); display(line); return 0; }
當上面的程式碼被編譯和執行時,它會產生下列結果:
呼叫建構函式 呼叫拷貝建構函式併為指標 ptr 分配記憶體 line 大小 : 10 釋放記憶體 釋放記憶體
下面的例項對上面的例項稍作修改,通過使用已有的同類型的物件來初始化新建立的物件:
例項
#include <iostream> using namespace std; class Line { public: int getLength( void ); Line( int len ); // 簡單的建構函式 Line( const Line &obj); // 拷貝建構函式 ~Line(); // 解構函式 private: int *ptr; }; // 成員函式定義,包括建構函式 Line::Line(int len) { cout << "呼叫建構函式" << endl; // 為指標分配記憶體 ptr = new int; *ptr = len; } Line::Line(const Line &obj) { cout << "呼叫拷貝建構函式併為指標 ptr 分配記憶體" << endl; ptr = new int; *ptr = *obj.ptr; // 拷貝值 } Line::~Line(void) { cout << "釋放記憶體" << endl; delete ptr; } int Line::getLength( void ) { return *ptr; } void display(Line obj) { cout << "line 大小 : " << obj.getLength() <<endl; } // 程式的主函式 int main( ) { Line line1(10); Line line2 = line1; // 這裡也呼叫了拷貝建構函式 display(line1); display(line2); return 0; }
當上面的程式碼被編譯和執行時,它會產生下列結果:
呼叫建構函式 呼叫拷貝建構函式併為指標 ptr 分配記憶體 呼叫拷貝建構函式併為指標 ptr 分配記憶體 line 大小 : 10 釋放記憶體 呼叫拷貝建構函式併為指標 ptr 分配記憶體 line 大小 : 10 釋放記憶體 釋放記憶體 釋放記憶體