程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> 關於C++ >> C++拷貝結構函數(深拷貝與淺拷貝)詳解

C++拷貝結構函數(深拷貝與淺拷貝)詳解

編輯:關於C++

C++拷貝結構函數(深拷貝與淺拷貝)詳解。本站提示廣大學習愛好者:(C++拷貝結構函數(深拷貝與淺拷貝)詳解)文章只能為提供參考,不一定能成為您想要的結果。以下是C++拷貝結構函數(深拷貝與淺拷貝)詳解正文


關於通俗類型的對象來講,它們之間的復制是很簡略的,例如:
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的復制進程。就類對象而言,雷同類型的類對象是經由過程拷貝結構函數來完成全部復制進程的。上面舉例解釋拷貝結構函數的任務進程。

#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)。

當用一個已初始化過了的自界說類類型對象去初始化另外一個新結構的對象的時刻,拷貝結構函數就會被主動挪用。也就是說,當類的對象須要拷貝時,拷貝結構函數將會被挪用。以下情形都邑挪用拷貝結構函數:
一個對象以值傳遞的方法傳入函數體
一個對象以值傳遞的方法從函數前往
一個對象須要經由過程別的一個對象停止初始化。

假如在類中沒有顯式地聲明一個拷貝結構函數,那末,編譯器將會主動生成一個默許的拷貝結構函數,該結構函數完成對象之間的位拷貝。位拷貝又稱淺拷貝,前面將停止解釋。

自界說拷貝結構函數是一種優越的編程作風,它可以阻攔編譯器構成默許的拷貝結構函數,進步源碼效力。

淺拷貝和深拷貝
在某些狀態下,類內成員變量須要靜態開拓堆內存,假如實施位拷貝,也就是把對象裡的值完整復制給另外一個對象,如A=B。這時候,假如B中有一個成員變量指針曾經請求了內存,那A中的誰人成員變量也指向統一塊內存。這就湧現了成績:當B把內存釋放了(如:析構),這時候A內的指針就是野指針了,湧現運轉毛病。

深拷貝和淺拷貝可以簡略懂得為:假如一個類具有資本,當這個類的對象產生復制進程的時刻,資本從新分派,這個進程就是深拷貝,反之,沒有從新分派資本,就是淺拷貝。上面舉個深拷貝的例子。

#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;語句完成的。

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved