今天进行了新的学习。
目录
虚析构函数是在C++中用于处理继承层次结构中的资源释放的一种技术。它在析构函数前加上 virtual 关键字,以允许正确地销毁派生类对象,即使通过基类指针删除派生类对象也能保证资源正确释放。
以下是虚析构函数的一些关键概念和用法:
当基类指针指向一个派生类对象,并且通过该指针删除对象时,通常只会调用基类的析构函数。这可能导致派生类特有的资源(如动态分配的内存)不会得到正确释放,从而产生内存泄漏。
通过将析构函数声明为虚函数,可以确保在删除派生类对象时调用派生类的析构函数,从而正确释放派生类的资源。
virtual 关键字。- class Base {
- public:
- virtual ~Base() {
- // 析构函数的实现,通常为空
- }
- };
virtual 关键字,因为它继承了基类的虚析构函数。- class Derived : public Base {
- public:
- ~Derived() {
- // 派生类析构函数的实现,释放派生类特有的资源
- }
- };
- Base* basePtr = new Derived();
- delete basePtr; // 正确释放资源,调用派生类析构函数
注意,如果不使用虚析构函数,上述代码将只调用基类的析构函数,可能导致派生类资源泄漏。
- class Animal {
- public:
- virtual ~Animal() {
- cout << "⽗类的析构函数执⾏了" << endl;
- }
- };
- class Person: public Animal {
- public:
- int* n;
- Person() {
- n = new int(10);
- }
- ~Person() override {
- cout << "⼦类的析构函数执⾏了" << endl;
- if (n != nullptr) {
- delete n;
- n = nullptr;
- }
- }
- };
- int main() {
- Animal* animal = new Person();
- // 如果没有虚析构函数的话,这⾥通过animal来销毁空间,的确可以把Person开辟的堆空间给销毁掉
- // 但是,由于只会触发⽗类中的析构函数,因此⽆法将Person属性n开辟的堆空间给销毁掉,造成内存泄漏
- // 解决⽅案:将⽗类的析构函数作为虚析构函数,完成动态绑定
- delete animal;
- return 0;
- }