为什么需要虚析构函数?
- 虚析构函数允许在运⾏时根据对象的实际类型调⽤正确的析构函数,从⽽实现多态性。
- 如果基类的析构函数不是虚的,当通过基类指针删除指向派⽣类对象的对象时,只会调⽤基类的析构函数,⽽不会调⽤派⽣类的析构函数。这可能导致派⽣类的资源未被正确释放,造成内存泄漏。
构造函数在对象的创建阶段被调⽤,对象的类型在构造函数中已经确定。因此,构造函数调⽤不涉及多态性,也就是说,在对象的构造期间⽆法实现动态绑定。虚构造函数没有意义,因为对象的类型在构造过程中就已经确定,不需要动态地选择构造函数。
通过基类指针删除指向派⽣类对象的对象时,这一句话的意思不太好懂,其实就是在说通过基类指针删除派生类对象意味着使用基类的指针来调用delete操作符,以释放指向派生类对象的内存。
代码说明,Base* basePtr = new Derived();
这行代码的意思是创建一个指向派生类对象(Derived)的基类指针(Base*),当通过delete basePtr基类指针删除派生类对象时,如果基类的析构函数不是虚函数,那么通过基类指针删除派生类对象时,只会调用基类的析构函数,而不会调用派生类的析构函数。这可能导致派生类中分配的资源无法得到正确的释放,从而引发内存泄漏。
#include <iostream>
class Base {
public:
Base() { std::cout << "Base constructor called" << std::endl; }
virtual ~Base() { std::cout << "Base destructor called" << std::endl; }
};
class Derived : public Base {
public:
Derived() { std::cout << "Derived constructor called" << std::endl; }
~Derived() { std::cout << "Derived destructor called" << std::endl; }
};
int main() {
Base* basePtr = new Derived(); // 创建派生类对象,并用基类指针指向它
delete basePtr; // 通过基类指针删除派生类对象
return 0;
}