亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

全面解析C++中的析構函數(shù)

 更新時間:2016年01月22日 14:25:47   投稿:goldensun  
這篇文章主要介紹了C++中的析構函數(shù),為C++入門學習中的基礎知識,需要的朋友可以參考下

“析構函數(shù)”是構造函數(shù)的反向函數(shù)。在銷毀(釋放)對象時將調用它們。通過在類名前面放置一個波形符 (~) 將函數(shù)指定為類的析構函數(shù)。例如,聲明 String 類的析構函數(shù):~String()。
在 /clr 編譯中,析構函數(shù)在釋放托管和非托管資源方面發(fā)揮了特殊作用。
析構函數(shù)通常用于在不再需要某個對象時“清理”此對象。請考慮 String 類的以下聲明:

// spec1_destructors.cpp
#include <string.h>

class String {
public:
  String( char *ch ); // Declare constructor
  ~String();      // and destructor.
private:
  char  *_text;
  size_t sizeOfText;
};

// Define the constructor.
String::String( char *ch ) {
  sizeOfText = strlen( ch ) + 1;

  // Dynamically allocate the correct amount of memory.
  _text = new char[ sizeOfText ];

  // If the allocation succeeds, copy the initialization string.
  if( _text )
   strcpy_s( _text, sizeOfText, ch );
}

// Define the destructor.
String::~String() {
  // Deallocate the memory that was previously reserved
  // for this string.
  if (_text)
   delete[] _text;
}

int main() {
  String str("The piper in the glen...");
}

在前面的示例中,析構函數(shù) String::~String 使用 delete 運算符來動態(tài)釋放為文本存儲分配的空間。
聲明析構函數(shù)
析構函數(shù)是具有與類相同的名稱但前面是波形符 (~) 的函數(shù)
該語法的第一種形式用于在類聲明中聲明或定義的析構函數(shù);第二種形式用于在類聲明的外部定義的析構函數(shù)。
多個規(guī)則管理析構函數(shù)的聲明。析構函數(shù):

  • 不接受參數(shù)。
  • 無法指定任何返回類型(包括 void)。
  • 無法使用 return 語句返回值。
  • 無法聲明為 const、volatile 或 static。但是,可以為聲明為 const、volatile 或 static 的對象的析構調用它們。
  • 可以聲明為 virtual。通過使用虛擬析構函數(shù),無需知道對象的類型即可銷毀對象 - 使用虛函數(shù)機制調用該對象的正確析構函數(shù)。請注意,析構函數(shù)也可以聲明為抽象類的純虛函數(shù)。

使用構造函數(shù)

當下列事件之一發(fā)生時,將調用析構函數(shù):
使用 delete 運算符顯式解除分配了使用 new 運算符分配的對象。使用 delete 運算符解除分配對象時,將為“大多數(shù)派生對象”或為屬于完整對象,但不是表示基類的子對象的對象釋放內存。此“大多數(shù)派生對象”解除分配一定僅對虛擬析構函數(shù)有效。在類型信息與實際對象的基礎類型不匹配的多重繼承情況下,取消分配可能失敗。
具有塊范圍的本地(自動)對象超出范圍。
臨時對象的生存期結束。
程序結束,并且存在全局或靜態(tài)對象。
使用析構函數(shù)的完全限定名顯式調用了析構函數(shù)。(有關詳細信息,請參閱顯式析構函數(shù)調用。)
前面的列表中所述的情況將確保所有對象均可通過用戶定義的方法進行銷毀。
如果基類或數(shù)據(jù)成員有一個可訪問的析構函數(shù),并且派生類未聲明析構函數(shù),則編譯器將生成一個析構函數(shù)。此編譯器生成的析構函數(shù)將為派生類型的成員調用基類析構函數(shù)和析構函數(shù)。默認析構函數(shù)是公共的。

析構函數(shù)可以隨意調用類成員函數(shù)和訪問類成員數(shù)據(jù)。從析構函數(shù)調用虛函數(shù)時,調用的函數(shù)是當前正在銷毀的類的函數(shù)。
析構的順序
當對象超出范圍或被刪除時,其完整析構中的事件序列如下所示:
將調用該類的析構函數(shù),并且會執(zhí)行該析構函數(shù)的主體。
按照非靜態(tài)成員對象的析構函數(shù)在類聲明中的顯示順序的相反順序調用這些函數(shù)。用于這些成員的構造的可選成員優(yōu)化列表不影響構造或析構的順序。
非虛擬基類的析構函數(shù)以聲明的相反順序被調用。
虛擬基類的析構函數(shù)以聲明的相反順序被調用。

// order_of_destruction.cpp
#include <stdio.h>

struct A1   { virtual ~A1() { printf("A1 dtor\n"); } };
struct A2 : A1 { virtual ~A2() { printf("A2 dtor\n"); } };
struct A3 : A2 { virtual ~A3() { printf("A3 dtor\n"); } };

struct B1   { ~B1() { printf("B1 dtor\n"); } };
struct B2 : B1 { ~B2() { printf("B2 dtor\n"); } };
struct B3 : B2 { ~B3() { printf("B3 dtor\n"); } };

int main() {
  A1 * a = new A3;
  delete a;
  printf("\n");

  B1 * b = new B3;
  delete b;
  printf("\n");

  B3 * b2 = new B3;
  delete b2;
}

輸出:

A3 dtor
A2 dtor
A1 dtor

B1 dtor

B3 dtor
B2 dtor
B1 dtor

虛擬基類
按照與虛擬基類在定向非循環(huán)圖形中顯示的順序的相反順序調用這些虛擬基類的析構函數(shù)(深度優(yōu)先、從左到右、后序遍歷)。下圖描述了繼承關系圖。

2016122142248275.jpeg (107×121)

演示虛擬基類的繼承關系圖
下面列出了圖中顯示的類的類頭。

  • class A
  • class B
  • class C : virtual public A, virtual public B
  • class D : virtual public A, virtual public B
  • class E : public C, public D, virtual public B

為了確定 E 類型的對象的虛擬基類的析構順序,編譯器將通過應用以下算法來生成列表:

  1. 向左遍歷關系圖,并從關系圖中的最深點開始(在此示例中,為 E)。
  2. 執(zhí)行左移遍歷,直到訪問了所有節(jié)點。記下當前節(jié)點的名稱。
  3. 重新訪問上一個節(jié)點(向下并向右)以查明要記住的節(jié)點是否為虛擬基類。
  4. 如果記住的節(jié)點是虛擬基類,請瀏覽列表以查看是否已將其輸入。如果它不是虛擬基類,則將其忽略。
  5. 如果記住的節(jié)點尚未包含在列表中,請將其添加到列表的底部。
  6. 向上遍歷關系圖并沿下一個路徑向右遍歷。
  7. 轉到步驟 2。
  8. 在用完最后一個向上路徑時,請記下當前節(jié)點的名稱。
  9. 轉到步驟 3。
  10. 繼續(xù)執(zhí)行此過程,直到底部節(jié)點再次成為當前節(jié)點。

因此,對于 E 類,析構順序為:

  1. 非虛擬基類 E。
  2. 非虛擬基類 D。
  3. 非虛擬基類 C。
  4. 虛擬基類 B。
  5. 虛擬基類 A。

此過程將生成唯一項的有序列表。任何類名均不會出現(xiàn)兩次。在構造列表后,將以相反的順序遍歷該列表,并且將調用列表中每個類(從最后一個到第一個)的析構函數(shù)。

如果某個類中的構造函數(shù)或析構函數(shù)依賴于要先創(chuàng)建或保留更長時間的另一個組件(例如,如果 A 的析構函數(shù)(上圖中所示)依賴于執(zhí)行其代碼時仍存在 B),則構造或析構的順序特別重要,反之亦然。
繼承關系圖中各個類之間的這種相互依賴項本質上是危險的,因為稍后派生類可以更改最左邊的路徑,從而更改構造和析構的順序。
非虛擬基類
按照相反的順序(按此順序聲明基類名稱)調用非虛擬基類的析構函數(shù)??紤]下列類聲明:

class MultInherit : public Base1, public Base2
...

在前面的示例中,先于 Base2 的析構函數(shù)調用 Base1 的析構函數(shù)。
顯式析構函數(shù)調用
很少需要顯式調用析構函數(shù)。但是,對置于絕對地址的對象進行清理會很有用。這些對象通常使用采用位置參數(shù)的用戶定義的 new 運算符進行分配。 delete 運算符不能釋放該內存,因為它不是從自由存儲區(qū)分配的(有關詳細信息,請參閱 new 和 delete 運算符)。但是,對析構函數(shù)的調用可以執(zhí)行相應的清理。若要顯式調用 s 類的對象 String 的析構函數(shù),請使用下列語句之一:

s.String::~String();   // Nonvirtual call
ps->String::~String();  // Nonvirtual call

s.~String();    // Virtual call
ps->~String();   // Virtual call

可以使用對前面顯示的析構函數(shù)的顯式調用的表示法,無論類型是否定義了析構函數(shù)。這允許您進行此類顯式調用,而無需了解是否為此類型定義了析構函數(shù)。顯式調用析構函數(shù),其中未定義的析構函數(shù)無效。

相關文章

  • C語言解決字符串中插入和刪除某段字符串問題

    C語言解決字符串中插入和刪除某段字符串問題

    這篇文章主要介紹了C語言解決字符串中插入和刪除某段字符串問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-02-02
  • 使用C++遞歸求解跳臺階問題

    使用C++遞歸求解跳臺階問題

    這篇文章主要介紹了使用C++求解跳臺階問題的方法,通過遞歸算法來解決,不算難,文中給出了計算思路,需要的朋友可以參考下
    2016-02-02
  • Opencv 馬賽克和毛玻璃效果與圖片融合的實現(xiàn)

    Opencv 馬賽克和毛玻璃效果與圖片融合的實現(xiàn)

    這篇文章主要為大家詳細介紹了通過OpenCV實現(xiàn)馬賽克和毛玻璃濾鏡效果與圖片的融合,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-11-11
  • 6個變態(tài)的C語言Hello World程序

    6個變態(tài)的C語言Hello World程序

    這篇文章主要介紹了6個變態(tài)的C語言Hello World程序,需要的朋友可以參考下
    2016-05-05
  • C++中的六個函數(shù)

    C++中的六個函數(shù)

    本文給大家介紹了C++中的六個函數(shù),非常不錯,具有一定的參考借鑒價值,需要的朋友參考下吧
    2018-05-05
  • C語言撲克牌游戲示例

    C語言撲克牌游戲示例

    大家好,本篇文章主要講的是C語言撲克牌游戲示例,感興趣的同學趕快來看一看吧,對你有幫助的話記得收藏一下,方便下次瀏覽
    2021-12-12
  • 從頭學習C語言之二維數(shù)組

    從頭學習C語言之二維數(shù)組

    這篇文章主要為大家詳細介紹了C語言之二維數(shù)組,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助
    2022-01-01
  • Qt+OpenCV利用幀差法實現(xiàn)車輛識別

    Qt+OpenCV利用幀差法實現(xiàn)車輛識別

    所謂幀差法也就是對連續(xù)圖像幀做差分運算,其結果與定義好的閾值比較,若大于閾值則為運動目標值為1,否則值為0?。本文將利用幀差法實現(xiàn)車輛識別,感興趣的可以了解一下
    2022-08-08
  • C語言詳細分析講解多文件的程序設計

    C語言詳細分析講解多文件的程序設計

    所謂的C語言多文件編程就是,將代碼實現(xiàn)模塊化。比如說一個項目的一項功能放在一個一個文件里,然后將實現(xiàn)這個功能的函數(shù)放在一個c文件<BR>
    2022-04-04
  • C++?Boost?Spirit進階教程

    C++?Boost?Spirit進階教程

    Boost是為C++語言標準庫提供擴展的一些C++程序庫的總稱。Boost庫是一個可移植、提供源代碼的C++庫,作為標準庫的后備,是C++標準化進程的開發(fā)引擎之一,是為C++語言標準庫提供擴展的一些C++程序庫的總稱
    2022-11-11

最新評論