析构函数(destructor)是成员函数的一种,它的名字与类名相同,但前面要加 ~ ,没有参数和返回值。一个类有且仅有一个析构函数。如果定义类时没写析构函数,则编译器生成默认析构函数。如果定义了析构函数,则编译器不生成默认析构函数。 析构函数在对象消亡时即自动被调用。可以定义析构函数在对象消亡前做善后工作。例如,对象如果在生存期间用 new 运算符动态分配了内存,则在各处写 delete 语句以确保程序的每条执行路径都能释放这片内存是比较麻烦的事情。有了析构函数,只要在析构函数中调用 delete 语句,就能确保对象运行中用 new 运算符分配的空间在对象消亡时被释放。例如下面的程序: class String{private:char* p;public:String(int n);~String();};String::~String(){delete[] p;}String::String(int n){p = new char[n];} 只要对象消亡,就会引发析构函数的调用。下面的程序说明了析构函数起作用的一些情况。 #include<iostream>using namespace std;class CDemo {public:~CDemo() { //析构函数cout << "Destructor called"<<endl;}};int main() {CDemo array[2]; //构造函数调用2次CDemo* pTest = new CDemo; //构造函数调用delete pTest; //析构函数调用cout << "-----------------------" << endl;pTest = new CDemo[2]; //构造函数调用2次delete[] pTest; //析构函数调用2次cout << "Main ends." << endl;return 0;} Destructor called ----------------------- Destructor called Destructor called Main ends. Destructor called Destructor called 第一次析构函数调用发生在第 13 行,delete 语句使得第 12 行动态分配的 CDemo 对象消亡。 接下来的两次析构函数调用发生在第 16 行,delete 语句释放了第 15 行动态分配的数组,那个数组中有两个 CDemo 对象消亡。最后两次析构函数调用发生在 main 函数结束时,因第 11 行的局部数组变量 array 中的两个元素消亡而引发。 函数的参数对象以及作为函数返回值的对象,在消亡时也会引发析构函数调用。例如: #include <iostream>using namespace std;class CDemo {public:~CDemo() { cout << "destructor" << endl; }};void Func(CDemo obj) {cout << "func" << endl;}CDemo d1;CDemo Test() {cout << "test" << endl;return d1;}int main() {CDemo d2;Func(d2);Test();cout << "after test" << endl;return 0;} func destructor test destructor after test destructor destructor 程序共输出 destructor 四次:
编程帮,一个分享编程知识的公众号。跟着站长一起学习,每天都有进步。 通俗易懂,深入浅出,一篇文章只讲一个知识点。 文章不深奥,不需要钻研,在公交、在地铁、在厕所都可以阅读,随时随地涨姿势。 文章不涉及代码,不烧脑细胞,人人都可以学习。 |
|