C++拷貝構造函數(深拷貝,淺拷貝)

1. 拷貝構造函數

  • 對於普通類型的對象來說,它們之間的複製是很簡單的,例如:
int a=88;
int b=a; 
  • 而類對象與普通對象不同,類對象內部結構一般較爲複雜,存在各種成員變量。下面看一個類對象拷貝的簡單例子。
#include <iostream>
using namespace std;

class CExample {
private:
     int a;
public:
     CExample(int b)
     { a=b;}
     void Show ()
     {
        cout<<a<<endl;
    }
};

int main()
{
     CExample A(100);
     CExample B=A;
     B.Show ();
     return 0;
} 

運行程序,屏幕輸出100。從以上代碼的運行結果可以看出,系統爲對象B分配了內存並完成了與對象A的複製過程
就類對象而言,相同類型的類對象是通過拷貝構造函數來完成整個複製過程的

2. 舉例

下面舉例說明拷貝構造函數的工作過程。

#include <iostream>
using namespace std;

class CExample {
private:
    int a;
public:
    CExample(int b)
    { a=b;}

    CExample(const CExample& C)
    {
        a = C.a;
    }
    void Show ()
    {
        cout<<a<<endl;
    }
};

int main()
{
    CExample A(100);
    CExample B=A;
    B.Show ();
    return 0;
} 

CExample(const CExample& C)就是我們自定義的拷貝構造函數
可見,拷貝構造函數是一種特殊的構造函數,函數的名稱必須和類名稱一致,它的唯一的一個參數是本類型的一個引用變量,該參數是const類型,不可變的。例如:類X的拷貝構造函數的形式爲X(X& x)。

當用一個已初始化過了的自定義類類型對象去初始化另一個新構造的對象的時候,拷貝構造函數就會被自動調用。也就是說,當類的對象需要拷貝時,拷貝構造函數將會被調用。

① 調用拷貝構造函數的情況

  • 一個對象以值傳遞的方式傳入函數體
  • 一個對象以值傳遞的方式從函數返回
  • 一個對象需要通過另外一個對象進行初始化

如果在類中沒有顯式地聲明一個拷貝構造函數,那麼,編譯器將會自動生成一個默認的拷貝構造函數,該構造函數完成對象之間的位拷貝。位拷貝又稱淺拷貝,後面將進行說明。

自定義拷貝構造函數是一種良好的編程風格,它可以阻止編譯器形成默認的拷貝構造函數,提高源碼效率

3. 淺拷貝和深拷貝

  在某些狀況下,類內成員變量需要動態開闢堆內存,如果實行淺拷貝,也就是把對象裏的值完全複製給另一個對象,如A=B。這時,如果B中有一個成員變量指針已經申請了內存,那A中的那個成員變量也指向同一塊內存。這就出現了問題:當B把內存釋放了(如:析構),這時A內的指針就是野指針了,出現運行錯誤。

  深拷貝和淺拷貝可以簡單理解爲:如果一個類擁有資源,當這個類的對象發生複製過程的時候,資源重新分配,這個過程就是深拷貝,反之,沒有重新分配資源,就是淺拷貝。下面舉個深拷貝的例子。

ps: 深拷貝和淺拷貝體現在類中,當類中有指針變量時,涉及到動態開闢堆內存,如果是淺拷貝,淺拷貝的類中指針變量指向同一塊空間,當其中一個對象析構時,另一個對象的成員變量指針就會變成野指針,會出現運行錯誤;而深拷貝會進行資源重新分配。

#include <iostream>
using namespace std;
class CA
{
 public:
  CA(int b,char* cstr)
  {
   a=b;
   str=new char[b];
   strcpy(str,cstr);
  }
  CA(const CA& C)
  {
   a=C.a;
   str=new char[a]; //深拷貝
   if(str!=0)
    strcpy(str,C.str);
  }
  void Show()
  {
   cout<<str<<endl;
  }
  ~CA()
  {
   delete str;
  }
 private:
  int a;
  char *str;
};

int main()
{
 CA A(10,"Hello!");
 CA B=A;
 B.Show();
 return 0;
} 
  • 淺拷貝資源後在釋放資源的時候會產生資源歸屬不清的情況導致程序運行出錯

Test(Test &c_t)是自定義的拷貝構造函數,拷貝構造函數的名稱必須與類名稱一致,函數的形式參數是本類型的一個引用變量,且必須是引用。

當用一個已經初始化過了的自定義類類型對象去初始化另一個新構造的對象的時候,拷貝構造函數就會被自動調用,如果你沒有自定義拷貝構造函數的時候,系統將會提供給一個默認的拷貝構造函數來完成這個過程,上面代碼的複製核心語句就是通過Test(Test &c_t)拷貝構造函數內的p1=c_t.p1;語句完成的。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章