詳解C++編程中的析構(gòu)函數(shù)
C++析構(gòu)函數(shù)
創(chuàng)建對(duì)象時(shí)系統(tǒng)會(huì)自動(dòng)調(diào)用構(gòu)造函數(shù)進(jìn)行初始化工作,同樣,銷毀對(duì)象時(shí)系統(tǒng)也會(huì)自動(dòng)調(diào)用一個(gè)函數(shù)來進(jìn)行清理工作(例如回收創(chuàng)建對(duì)象時(shí)消耗的各種資源),這個(gè)函數(shù)被稱為析構(gòu)函數(shù)。
析構(gòu)函數(shù)(Destructor)也是一種特殊的成員函數(shù),沒有返回值,不需要用戶調(diào)用,而是在銷毀對(duì)象時(shí)自動(dòng)執(zhí)行。與構(gòu)造函數(shù)不同的是,析構(gòu)函數(shù)的名字是在類名前面加一個(gè)”~“符號(hào)。
注意:析構(gòu)函數(shù)沒有參數(shù),不能被重載,因此一個(gè)類只能有一個(gè)析構(gòu)函數(shù)。如果用戶沒有定義,那么編譯器會(huì)自動(dòng)生成。
析構(gòu)函數(shù)舉例:
#include <iostream> using namespace std; class Student{ private: char *name; int age; float score; public: //構(gòu)造函數(shù) Student(char *, int, float); //析構(gòu)函數(shù) ~Student(); //普通成員函數(shù) void say(); }; Student::Student(char *name1, int age1, float score1):name(name1), age(age1), score(score1){} Student::~Student(){ cout<<name<<"再見"<<endl; } void Student::say(){ cout<<name<<"的年齡是 "<<age<<",成績(jī)是 "<<score<<endl; } int main(){ Student stu1("小明", 15, 90.5f); stu1.say(); Student stu2("李磊", 16, 95); stu2.say(); Student stu3("王爽", 16, 80.5f); stu3.say(); cout<<"main 函數(shù)即將運(yùn)行結(jié)束"<<endl; return 0; }
運(yùn)行結(jié)果:
小明的年齡是 15,成績(jī)是 90.5 李磊的年齡是 16,成績(jī)是 95 王爽的年齡是 16,成績(jī)是 80.5 main 函數(shù)即將運(yùn)行結(jié)束 王爽再見 李磊再見 小明再見
可以看出,析構(gòu)函數(shù)在 main 函數(shù)運(yùn)行結(jié)束前被執(zhí)行,并且調(diào)用順序和構(gòu)造函數(shù)正好相反,為了方便記憶,我們可以將之理解為一個(gè)棧,先入后出。
析構(gòu)函數(shù)的執(zhí)行順序?yàn)槭裁词欠吹摹?br />
析構(gòu)函數(shù)在對(duì)象被銷毀前執(zhí)行;要知道析構(gòu)函數(shù)什么時(shí)候被調(diào)用,就要先知道對(duì)象什么時(shí)候被銷毀。
對(duì)象可以認(rèn)為是通過類這種數(shù)據(jù)類型定義的變量,它的很多特性和普通變量是一樣的,例如作用域、生命周期等。由此可以推斷,對(duì)象這種變量的銷毀時(shí)機(jī)和普通變量是一樣的。
總結(jié)起來,有下面幾種情況:
1) 如果在一個(gè)函數(shù)中定義了一個(gè)對(duì)象(auto 局部變量),當(dāng)這個(gè)函數(shù)運(yùn)行結(jié)束時(shí),對(duì)象就會(huì)被銷毀,在對(duì)象被銷毀前自動(dòng)執(zhí)行析構(gòu)函數(shù)。
2) static 局部對(duì)象在函數(shù)調(diào)用結(jié)束時(shí)并不銷毀,因此也不調(diào)用析構(gòu)函數(shù),只有在程序結(jié)束時(shí)(如 main 函數(shù)結(jié)束或調(diào)用 exit 函數(shù))才調(diào)用 static 局部對(duì)象的析構(gòu)函數(shù)。
3) 如果定義了一個(gè)全局對(duì)象,也只有在程序結(jié)束時(shí)才會(huì)調(diào)用該全局對(duì)象的析構(gòu)函數(shù)。
4) 如果用 new 運(yùn)算符動(dòng)態(tài)地建立了一個(gè)對(duì)象,當(dāng)用 delete 運(yùn)算符釋放該對(duì)象時(shí),先調(diào)用該對(duì)象的析構(gòu)函數(shù)。
注意:析構(gòu)函數(shù)的作用并不是刪除對(duì)象,而是在撤銷對(duì)象占用的內(nèi)存之前完成一些清理工作,使這部分內(nèi)存可以分配給新對(duì)象使用。
C++調(diào)用構(gòu)造函數(shù)和析構(gòu)函數(shù)的順序
在使用構(gòu)造函數(shù)和析構(gòu)函數(shù)時(shí),需要特別注意對(duì)它們的調(diào)用時(shí)間和調(diào)用順序。在一般情況下,調(diào)用析構(gòu)函數(shù)的次序正好與調(diào)用構(gòu)造函數(shù)的次序相反:最先被調(diào)用的構(gòu)造函數(shù),其對(duì)應(yīng)的(同一對(duì)象中的)析構(gòu)函數(shù)最后被調(diào)用,而最后被調(diào)用的構(gòu)造函數(shù),其對(duì)應(yīng)的析構(gòu)函數(shù)最先被調(diào)用。如例9.5所示,先執(zhí)行stud2的析構(gòu)函數(shù),再執(zhí)行stu1的析構(gòu)函數(shù)。
可以簡(jiǎn)記為:先構(gòu)造的后析構(gòu),后構(gòu)造的先析構(gòu),它相當(dāng)于一個(gè)棧,先進(jìn)后出。
但是,并不是在任何情況下都是按這一原則處理的。對(duì)象可以在不同的作用域中定義,可以有不同的存儲(chǔ)類別。這些會(huì)影響調(diào)用構(gòu)造函數(shù)和析構(gòu)函數(shù)的時(shí)機(jī)。
下面歸納一下什么時(shí)候調(diào)用構(gòu)造函數(shù)和析構(gòu)函數(shù):
1) 在全局范圍中定義的對(duì)象(即在所有函數(shù)之外定義的對(duì)象),它的構(gòu)造函數(shù)在文件中的所有函數(shù)(包括main函數(shù))執(zhí)行之前調(diào)用。但如果一個(gè)程序中有多個(gè)文件,而不同的文件中都定義了全局對(duì)象,則這些對(duì)象的構(gòu)造函數(shù)的執(zhí)行順序是不確定的。當(dāng)main函數(shù)執(zhí)行完畢或調(diào)用exit函數(shù)時(shí)(此時(shí)程序終止),調(diào)用析構(gòu)函數(shù)。
2) 如果定義的是局部自動(dòng)對(duì)象(例如在函數(shù)中定義對(duì)象),則在建立對(duì)象時(shí)調(diào)用其構(gòu)造函數(shù)。如果函數(shù)被多次調(diào)用,則在每次建立對(duì)象時(shí)都要調(diào)用構(gòu)造函數(shù)。在函數(shù)調(diào)用結(jié)束、對(duì)象釋放時(shí)先調(diào)用析構(gòu)函數(shù)。
3) 如果在函數(shù)中定義靜態(tài)(static )局部對(duì)象,則只在程序第一次調(diào)用此函數(shù)建立對(duì)象時(shí)調(diào)用構(gòu)造函數(shù)一次,在調(diào)用結(jié)束時(shí)對(duì)象并不釋放,因此也不調(diào)用析構(gòu)函數(shù),只在main函數(shù)結(jié)束或調(diào)用exit函數(shù)結(jié)束程序時(shí),才調(diào)用析構(gòu)函數(shù)。
例如,在一個(gè)函數(shù)中定義了兩個(gè)對(duì)象:
void fn(){ Student stud1; //定義自動(dòng)局部對(duì)象 static Student stud2; //定義靜態(tài)局部對(duì)象 }
在調(diào)用fn函數(shù)時(shí),先調(diào)用stud1的構(gòu)造函數(shù),再調(diào)用stud2的構(gòu)造函數(shù),在fn調(diào)用結(jié)束時(shí),stud1是要釋放的(因?yàn)樗亲詣?dòng)局部對(duì)象),因此調(diào)用stud1的析構(gòu)函數(shù)。而stud2 是靜態(tài)局部對(duì)象,在fn調(diào)用結(jié)束時(shí)并不釋放,因此不調(diào)用stud2的析構(gòu)函數(shù)。直到程序結(jié)束釋放stud2時(shí),才調(diào)用stud2的析構(gòu)函數(shù)??梢钥吹絪tud2是后調(diào)用構(gòu)造函數(shù)的,但并不先調(diào)用其析構(gòu)函數(shù)。原因是兩個(gè)對(duì)象的存儲(chǔ)類別不同、生命周期不同。
- 淺談C++基類的析構(gòu)函數(shù)為虛函數(shù)
- 簡(jiǎn)要解讀C++的動(dòng)態(tài)和靜態(tài)關(guān)聯(lián)以及虛析構(gòu)函數(shù)
- 簡(jiǎn)單介紹C++編程中派生類的析構(gòu)函數(shù)
- C++類成員構(gòu)造函數(shù)和析構(gòu)函數(shù)順序示例詳細(xì)講解
- c++基礎(chǔ)語法:構(gòu)造函數(shù)與析構(gòu)函數(shù)
- 深入解析C++中的構(gòu)造函數(shù)和析構(gòu)函數(shù)
- C++虛析構(gòu)函數(shù)的使用分析
- 解析C++中虛析構(gòu)函數(shù)的作用
- 深入C++中構(gòu)造函數(shù)、拷貝構(gòu)造函數(shù)、賦值操作符、析構(gòu)函數(shù)的調(diào)用過程總結(jié)
- 從匯編看c++的默認(rèn)析構(gòu)函數(shù)的使用詳解
- C++函數(shù)返回值為對(duì)象時(shí),構(gòu)造析構(gòu)函數(shù)的執(zhí)行細(xì)節(jié)
- 全面解析C++中的析構(gòu)函數(shù)
相關(guān)文章
C語言學(xué)習(xí)之條件和?if...else語句詳解
這篇文章主要給大家介紹了C語言中的條件和?if...else語句,文中通過代碼示例給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2023-12-12C/C++?Qt?數(shù)據(jù)庫與TableView實(shí)現(xiàn)多組件聯(lián)動(dòng)
Qt?數(shù)據(jù)庫組件與TableView組件實(shí)現(xiàn)聯(lián)動(dòng)效果,本文通過案例給大家講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2021-12-12C語言實(shí)現(xiàn)小學(xué)生考試系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)小學(xué)生考試系統(tǒng),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-03-03