C 學習篇 拷貝建構函式

2021-04-25 09:13:23 字數 2857 閱讀 4647

在學習這一章內容前我們已經學習過了類的建構函式和析構函式的相關知識,對於普通型別的物件來說,他們之間的複製是很簡單的,例如:

int a = 10;

int b =a;

自己定義的類的物件同樣是物件,誰也不能阻止我們用以下的方式進行複製,例如:

#include

using namespace std; 

class test 

protected: 

int p1; 

}; 

void main() 

普通物件和類物件同為物件,他們之間的特性有相似之處也有不同之處,類物件內部存在成員變數,而普通物件是沒有的,當同樣的複製方法發生在不同的物件上的時候,那麼系統對他們進行的操作也是不一樣的,就類物件而言,相同型別的類物件是通過拷貝建構函式來完成整個複製過程的,在上面的**中,我們並沒有看到拷貝建構函式,同樣完成了複製工作,這又是為什麼呢?因為當乙個類沒有自定義的拷貝建構函式的時候系統會自動提供乙個預設的拷貝建構函式,來完成複製工作。

下面,我們為了說明情況,就普通情況而言(以上面的**為例),我們來自己定義乙個與系統預設拷貝建構函式一樣的拷貝建構函式,看看它的內部是如何工作的!

**如下:

#include

using namespace std; 

class test 

test(test &c_t)//這裡就是自定義的拷貝建構函式 

執行結果:載入建構函式

載入copy建構函式

中國軟體開發工作室:www.***ev-lab.com中國軟體開發實驗室

載入copy建構函式

載入test建構函式

載入建構函式!

載入建構函式!

載入建構函式!

上面**就演示了深拷貝的問題,對物件b的cname屬性採取了新開闢記憶體的方式避免了記憶體歸屬不清所導致析構釋放空間時候的錯誤,最後我必須提一下,對於上面的程式我的解釋並不多,就是希望讀者本身執行程式觀察變化,進而深刻理解。

拷貝和淺拷貝的定義可以簡單理解成:如果乙個類擁有資源(堆,或者是其它系統資源),當這個類的物件發生複製過程的時候,這個過程就可以叫做深拷貝,反之物件存在資源但複製過程並未複製資源的情況視為淺拷貝。

淺拷貝資源後在釋放資源的時候會產生資源歸屬不清的情況導致程式執行出錯,這點尤其需要注意!

以前我們的教程中討論過函式返回物件產生臨時變數的問題,接下來我們來看一下在函式中返回自定義型別物件是否也遵循此規則產生臨時物件!

先執行下列**:

#include

using namespace std; 

class internet 

;  internet(char *name,char *address) 

void main() 

執行結果:載入建構函式

載入copy建構函式

載入建構函式!

載入建構函式!

載入建構函式!

從上面的**執行結果可以看出,程式一共載入過析構函式三次,證明了由函式返回自定義型別物件同樣會產生臨時變數,事實上物件a得到的就是這個臨時internet類型別物件temp的值。

這一下節的內容我們來說一下無名物件。

利用無名物件初始化物件系統不會不呼叫拷貝建構函式。

那麼什麼又是無名物件呢?

很簡單,如果在上面程式的main函式中有:

internet ("中國軟體開發實驗室","www.***ev-lab.com");

這樣的一句語句就會產生乙個無名物件,無名物件會呼叫建構函式但利用無名物件初始化物件系統不會不呼叫拷貝建構函式!

下面三段**是很見到的三種利用無名物件初始化物件的例子。

#include

using namespace std; 

class internet 

{  public: 

internet(char *name,char *address) 

{  cout<<"載入建構函式"cout《執行結果:載入建構函式

中國軟體開發實驗室

載入建構函式!

上面**的執行結果有點「出人意料」,從思維邏輯上說,當無名物件建立了後,是應該呼叫自定義拷貝建構函式,或者是預設拷貝建構函式來完成複製過程的,但事實上系統並沒有這麼做,因為無名物件使用過後在整個程式中就失去了作用,對於這種情況c++會把**看成是:

internet a("中國軟體開發實驗室",www.***ev-lab.com);

省略了建立無名物件這一過程,所以說不會呼叫拷貝建構函式。

最後讓我們來看看引用無名物件的情況。

#include

using namespace std;   

class internet   

{   

public:   

internet(char *name,char *address)   

{   

cout<<"載入建構函式"cout<

執行結果:載入建構函式

中國軟體開發實驗室

載入建構函式!

引用本身是物件的別名,和複製並沒有關係,所以不會呼叫拷貝建構函式,但要注意的是,在c++看來:

internet &a=internet("中國軟體開發實驗室","www.***ev-lab.com");

是等價與:

internet a("中國軟體開發實驗室","www.***ev-lab.com");

的,注意觀察呼叫析構函式的位置(這種情況是在main()外呼叫,而無名物件本身是在main()內析構的)。

文章出處:http://www.diybl.com/course/3_program/c++/cppsl/20081117/151284_2.html

C 建構函式 拷貝建構函式

建構函式 class base private int m var 建構函式無返回值型別,函式名和型別相同。拷貝建構函式傳遞引數為引用。1 class base2 7 拷貝建構函式 8 base base ref m var ref m var 9 11 private 12 intm var 13...

C 學習筆記 拷貝建構函式

拷貝建構函式是一種特殊的建構函式 1 它是建構函式,所以函式名是類名 沒有返回值 2 它是特殊的建構函式 引數形式是固定的 class object 拷貝建構函式的含義 以乙個物件為藍本,來構造另乙個物件。object b object a b 稱作 以b為藍本,建立乙個新的物件a。a是b的乙個拷貝...

C 學習(4) 拷貝建構函式

先來看c 物件的賦值 student xiaoming new student 一年級 一班 student xiaohong xiaoming c 支援物件的賦值,上面的 是沒有問題的,物件xiaohong擁有和物件xiaoming一樣的屬性 同樣是 一年級一班 問題1 以上 物件的賦值是怎麼實現...