C++ 的構造/析構/賦值/拷貝函數比較

來源:互聯網
上載者:User
 建構函式、解構函式與賦值函數是每個類最基本的函數。每個類只有一個解構函式,但可以有多個建構函式(包含一個拷貝建構函式,其它的稱為普通建構函式)和 多個賦值函數(除了同類的賦值以外,還有其他的賦值方法)。對於任意一個類A,如果不想編寫上述函數,C++編譯器將自動為A產生四個預設的函數,如
A(void);                    // 預設的無參數建構函式
A(const A &a);              // 預設的拷貝建構函式
~A(void);                   // 預設的解構函式
A & operate =(const A &a); // 預設的賦值函數

有幾個需要注意的內容:
@ 建構函式與解構函式的另一個特別之處是沒有傳回值類型
@ 構造從類層次的最頂層的基類開始,在每一層中,首先調用基類的建構函式,然後調用成員對象的建構函式。析構則嚴格按照與構造相反的次序執行,在析構的時候,最低層的衍生類別的解構函式最開始被調用,然後調用每個基類的解構函式。
@ “預設的拷貝建構函式”和“預設的賦值函數”均採用“位拷貝”而非“值拷貝”的方式來實現,倘若類中含有指標變數,這兩個函數註定將出錯

下面通過例子進一步說明,

1.建構函式的初始化表
設存在兩個類:

class A
{
     …
     A(void);                // 無參數建構函式
     A(const A &other);      // 拷貝建構函式
     A & operate =( const A &other);  // 賦值函數
    virtual ~A(void);        //解構函式
};
class B
{
public:
     B(const A &a);    // B的建構函式

private:   
     A   m_a;            // 成員對象
};

下面面是B的建構函式的2個實現,其中第一個的類B的建構函式在其初始化表裡調用了類A的拷貝建構函式,從而將成員對象m_a初始化;而第二個的B的構造 函數在函數體內用賦值的方式將成員對象m_a初始化。我們看到的只是一條指派陳述式,但實際上B的建構函式幹了兩件事:先暗地裡建立m_a對象(調用了A的 無參數建構函式),再調用類A的賦值函數,將參數a賦給m_a。

B::B(const A &a)
: m_a(a)
{
    …
}

B::B(const A &a)
{
     m_a = a;
     …
}

2.拷貝函數和建構函式的區別
拷貝建構函式是在對象被建立時調用的,而賦值函數只能被已經存在了的對象調用。
String a(“hello”);
String b(“world”);
String c = a; // 調用了拷貝建構函式,最好寫成 c(a);
c = b;            // 調用了賦值函數
本例中第三個語句的風格較差,宜改寫成String c(a) 以區別於第四個語句。

如果我們實在不想編寫拷貝建構函式和賦值函數,又不允許別人使用編譯器產生的預設函數,可以將拷貝建構函式和賦值函式宣告為私人函數,不用編寫代碼。

3.解構函式與虛解構函式
基類的建構函式、解構函式、賦值函數都不能被衍生類別繼承。如果類之間存在繼承關係,在編寫上述基本函數時應注意以下事項:
@ 衍生類別的建構函式應在其初始化表裡調用基類的建構函式
@ 基類與衍生類別的解構函式應該為虛(即加virtual關鍵字)

#include <iostream>
class Base
{
public:
    virtual ~Base() { cout<< "~Base" << endl ; }
};
class Derived : public Base
{
public:
    virtual ~Derived() { cout<< "~Derived" << endl ; }
};

void main(void)
{
     Base * pB = new Derived;  // upcast
    delete pB;
}

輸出結果為:
       ~Derived
       ~Base
如果解構函式不為虛,那麼輸出結果為
       ~Base

樣本說明:

編寫類String的建構函式,解構函式,賦值函數.

class String{

     public:

            String(const char *str=NULL);

            String(const String &other);

            ~String(void);

            String& operate = (const String &other);

     private:

            char * m_data;

};

String::String(const char*str){

        if(str==NULL){

             m_data = new char[1];

             *m_data ='/0';

          }

          else{

               int length = strlen(str);

               m_data= new char[length+1];

               strcpy(m_data,str);

           }

}

String::String(const String &other){

           int length = strlen(other.m_data);

           m_data = new char[length+1];

           strcpy(m_data,other.m_data);

}

String::~String(void){

          delete [] m_data;

}

String & String::operate = (const String &other){

          if(this==&other)  //檢查自賦值.

                 return *this;

           delete [] m_data; //釋放原有的記憶體資源.

           int length = strlen(other.m_data);

           m_data = new char[length+1];

           strcpy(m_data,other.m_data);

            return *this;

}

    

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.