歡迎來到Linux教程網
Linux教程網
Linux教程網
Linux教程網
您现在的位置: Linux教程網 >> UnixLinux >  >> Linux編程 >> Linux編程

C++ 析構函數&虛析構函數

1.為什麼要定義虛析構函數?

如果有一個帶有虛函數功能的類,則它需要一個虛析構函數,原因如下:
1)如果一個類有虛函數功能,它經常作為一個基類使用;
2)如果它是一個基類,它的派生類經常使用new來分配;
3)如果一個派生類對象使用new來分配,並且通過一個指向它的基類的指針來控制,那麼它經常通過一個指向它的基類的指針來刪除它(如果基類沒有虛析構函數,結果將是不確定的,實際發生時,派生類的析構函數永遠不會被調用);
基類有虛析構函數的話,最底層的派生類的析構函數最先被調用,然後各個基類的析構函數被調用。

2.聲明為保護(protected)的析構函數

如果在一個派生類中定義了基類以外的成員對象,且基類析構函數不是virtual修飾的,那麼當基類指針或引用指向派生類對象並析構(例如自動對象在函數作用域結束時;或者通過delete)時,會調用基類的析構函數而導致派生類定義的成員沒有被析構,產生內存洩露等問題。
雖然把析構函數定義成virtual的可以解決這個問題,但是當其它成員函數都不是virtual函數時(即基類中沒有虛函數),會在基類和派生類引入vtable,實例引入vptr造成運行時的性能損失。如果確定不需要直接而是只通過派生類對象使用基類,
可以把析構函數定義為protected(這樣會導致基類和派生類外使用自動對象和delete時的錯誤,因為訪問權限禁止調用析構函數),就不會導致以上問題。

3.構造函數、析構函數聲明為私有和保護時的用法

從語法上來講,一個函數被聲明為protected或者private,那麼這個函數就不能從“外部”直接被調用了。
對於protected的函數,子類的“內部”的其他函數可以調用之;
而對於private的函數,只能被本類“內部”的其他函數說調用;

通常使用的場景如下:
1)如果你不想讓外面的用戶直接構造一個類(假設這個類的名字為A)的對象,而希望用戶只能構造這個類A的子類,那你就可以將類A的構造函數/析構函數聲明為protected,而將類A的子類的構造函數/析構函數聲明為public。

例如:

class A
{
  protected: A(){}
  public:
  ....
};
calss B : public A
{
  public: B(){}
  ....
};

A a; // error
B b; // ok

2)如果將構造函數/析構函數聲明為private,那只能這個類的“內部”的函數才能構造這個類的對象了。
例如:

class A
{
  private:
    A(){  }
    ~A(){ }

  public:
    void Instance()//類A的內部的一個函數
    {
        A a;
    }
};

上面的代碼是能通過編譯的。上面代碼裡的Instance函數就是類A的內部的一個函數。Instance函數體裡就構建了一個A的對象。
但是,這個Instance函數還是不能夠被外面調用的。為什麼呢?
如果要調用Instance函數,必須有一個對象被構造出來。但是構造函數被聲明為private的了。外部不能直接構造一個對象出來。

A aObj; // 編譯通不過
aObj.Instance();

但是,如果Instance是一個static靜態函數的話,就可以不需要通過一個對象,而可以直接被調用。

class A
{
  private:
    A():data(10){ cout << "A" << endl; }
    ~A(){ cout << "~A" << endl; }

  public:
    static A& Instance()
    {
        static A a;
        return a;
    }
 
    void Print()
    {
        cout << data << endl;
    }

  private:
    int data;
};

A& ra = A::Instance();
ra.Print();

上面的代碼其實是設計模式singleton模式的一個簡單的C++代碼實現。

C++ 設計新思維》 下載見 http://www.linuxidc.com/Linux/2014-07/104850.htm

C++ Primer Plus 第6版 中文版 清晰有書簽PDF+源代碼 http://www.linuxidc.com/Linux/2014-05/101227.htm

讀C++ Primer 之構造函數陷阱 http://www.linuxidc.com/Linux/2011-08/40176.htm

讀C++ Primer 之智能指針 http://www.linuxidc.com/Linux/2011-08/40177.htm

讀C++ Primer 之句柄類 http://www.linuxidc.com/Linux/2011-08/40175.htm

將C語言梳理一下,分布在以下10個章節中:

  1. Linux-C成長之路(一):Linux下C編程概要 http://www.linuxidc.com/Linux/2014-05/101242.htm
  2. Linux-C成長之路(二):基本數據類型 http://www.linuxidc.com/Linux/2014-05/101242p2.htm
  3. Linux-C成長之路(三):基本IO函數操作 http://www.linuxidc.com/Linux/2014-05/101242p3.htm
  4. Linux-C成長之路(四):運算符 http://www.linuxidc.com/Linux/2014-05/101242p4.htm
  5. Linux-C成長之路(五):控制流 http://www.linuxidc.com/Linux/2014-05/101242p5.htm
  6. Linux-C成長之路(六):函數要義 http://www.linuxidc.com/Linux/2014-05/101242p6.htm
  7. Linux-C成長之路(七):數組與指針 http://www.linuxidc.com/Linux/2014-05/101242p7.htm
  8. Linux-C成長之路(八):存儲類,動態內存 http://www.linuxidc.com/Linux/2014-05/101242p8.htm
  9. Linux-C成長之路(九):復合數據類型 http://www.linuxidc.com/Linux/2014-05/101242p9.htm
  10. Linux-C成長之路(十):其他高級議題

Copyright © Linux教程網 All Rights Reserved