文章目录
析构函数的概念
析构函数与构造函数功能相反,析构函数类似于Destroy的功能。
在此,不要搞混了析构函数的功能,它不是用来销毁对象本身的。因为局部对象储存在栈帧里,函数结束时,栈帧就销毁了,存在在栈帧里的对象也就被释放了,不需要我们去释放。C++规定对象在销毁时 会自动调用析构函数,完成对象中资源的清理释放工作。
严格来说,Data类的是不需要析构函数的【它没有资源需要清理】,栈才需要。简单理解的话就是,一般情况下,显示申请了资源(也可以说是开空间,比如:栈)才需要自己实现析构,其余情况不需要显示写。
析构函数的特点
- 析构函数名:在类名前面加~。 (~在C语言中有按位取反的意思)
- 无参数无返回值。(同样析构函数不需要写void)(无参数意味着析构函数没有函数重载)
- 一个类只有一个析构函数(和无参有关)。若未显式定义,系统会自动生成默认的析构函数。
- 对象生命周期结束时,系统自动调用析构函数。
- 跟构造函数类似,我们不写编译器自动生成的析构函数对内置类型成员不做处理,自定类型成员会调用他的析构函数
还需要注意的是我们显示写析构函数,对于自定义类型成员也会调用他的析构,也就是说自定义类型成员无论什么情况都会自动调用析构函数。
cpp
#include<iostream>
using namespace std;
typedef int STDataType;
class Stack
{
public:
//构造函数
Stack(int n = 4)
{
_a = (STDataType*)malloc(sizeof(STDataType) * n);
if (nullptr == _a)
{
perror("malloc申请空间失败");
return;
}
_capacity = n;
_top = 0;
}
//析构函数
~Stack()
{
cout << "~Stack()" << endl;//没有任何作用,这样只是为了能让我们知道调用了析构函数
free(_a);
_a = nullptr;
_top = _capacity = 0;
}
private:
STDataType* _a;
size_t _capacity;
size_t _top;
};
class MyQueue
{
public:
private:
Stack pushst; //这个是上面我们自定义类型Stack
//编译器默认⽣成MyQueue的析构函数调⽤了Stack的析构,释放的Stack内部的资源
// 显⽰写析构,也会⾃动调⽤Stack的析构
//~MyQueue(){}
Stack popst;
};
int main()
{
Stack st;
return 0;
}
- 是否写析构函数:
-
如果类中没有申请资源时,析构函数可以不写,直接使用编译器生成的默认析构函数,如Date;它的里面只有一些变量
-
如果默认生成的析构就可以用,也就不需要显示写析构,如MyQueue;它会调Stack的析构
-
但是有资源申请时,一定要自己写析构,否则会造成资源泄漏,如Stack中malloc了空间。
- 一个局部域的多个对象,C++规定后定义的,需要先析构
C++和C对比(之前的两个栈实现一个队列)
不会再忘记调用Init和Destory函数了