为何需要虚析构函数

探讨当基类指针指向派生类对象并调用delete时,仅执行基类析构函数的问题,及其解决方法——虚析构函数的重要性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

派生类对象传递给基类参数时,并通过该参数调用函数时,将执行基类的函数。然而,还存在一个问题:如果基类指针指向的是派生类对象,通过该指针调用运算符delete时,结果将会如何呢?

在这里插入图片描述
运行结果:
在这里插入图片描述
注意⚠️:
由于使用了关键字new,在自由存储区中构造函了Tuna和Fish,但delete没有调用Tuna的析构函数,而只调用了Fish的析构函数;而构造和析构的局部变量tuna时,调用了基类的和派生类的构造函数和析构函数。派生类对象的构造和析构过程;它表明在析构过程中,需要调用相关的析构函数,包括~Tuna();显然是什么地方出问题了。
对于使用new在自由存储区的实例化的派生类对象,如果将其赋给基类指针,并通过该指针调用delet,将不会调用派生类的析构函数。这可能导致资源未释放,内存泄漏等问题。
要避免这种问题,可将析构函数声明为虚函数。确保基类指针调用delete时,将调用派生类的析构函数。

虚析构函数是一个在基类中定义的虚函数,它在继承类对象销毁时被调用。用 `virtual` 关键字声明的函数就是虚析构函数。 需要虚析构函数的原因是当一个基类指针指向一个派生类对象时,如果基类中的函数不是虚函数,那么在删除基类指针时,只会调用基类的函数,而不会调用派生类的函数,导致派生类对象的内存空间没有被正确释放,从而导致内存泄漏。 举个例子,假设有一个基类 `Animal` 和一个派生类 `Dog`,它们都有一个 `DoSomething` 函数和一个函数。如果 `Animal` 的函数不是虚函数,那么在使用基类指针删除 `Dog` 对象时,只会调用 `Animal` 的函数,而不会调用 `Dog` 的函数,导致 `Dog` 对象的内存没有被正确释放。 因此,为了确保在使用基类指针指向派生类对象时能够正确调用派生类的函数,需要在基类中声明虚析构函数。这样,在使用基类指针删除派生类对象时,会首先调用派生类的函数,然后再调用基类的函数,从而正确释放对象的内存空间。 以下是一个基类和派生类的示例代码,演示了虚析构函数的用法: ```cpp class Animal { public: virtual ~Animal() { cout << "Animal 被销毁" << endl; } virtual void DoSomething() { cout << "Animal 看看周围" << endl; } }; class Dog : public Animal { public: ~Dog() { cout << "Dog 被销毁" << endl; } void DoSomething() { cout << "Dog 摇摇尾巴" << endl; } }; ``` 在上面的代码中,`Animal` 类中的函数被声明为虚析构函数,而 `Dog` 类中的函数重载了 `Animal` 类的虚析构函数。这样,当使用基类指针指向 `Dog` 类对象并使用 `delete` 运算符删除该指针时,首先会调用 `Dog` 类的函数,然后再调用 `Animal` 类的函数,从而正确释放对象的内存空间。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值