虚析构函数能防止多态内存泄漏,因为通过基类指针 delete 派生类对象时,它确保先调用派生类析构函数再调用基类析构函数,从而正确释放派生类中申请的资源。
当用基类指针指向派生类对象并 delete 时,若基类析构函数不是虚函数,C++ 只会调用基类的析构函数,派生类中申请的资源(如 new 出的内存、打开的文件句柄等)不会被释放,直接导致内存泄漏或资源泄露。
虚析构函数的作用就是让析构行为也支持动态绑定:哪怕通过基类指针删除对象,也能正确触发派生类析构函数 → 基类析构函数 → 逐层向上完成清理。
virtual ~Base() = 0; // 声明
Base::~Base() {} // 定义,不可省略
典型错误代码:
class Base {
public:
~Base() { std::cout << "Base dtor\n"; }
};
class Derived : public Base {
int* data = new int[100];
public:
~Derived() { delete[] data; std::cout << "Derived dtor\n"; }
};
Base* p = new Derived();
delete p; // 只输出 "Base dtor",data 泄露!
运行后不会报错,但 data 指向的 100 个 int 内存永远无法回收。这种泄漏在长期运行服务或频繁创建销毁对象时会逐渐耗尽内存。
-Wnon-virtual-dtor 警告(建议开启)cppcoreguidelines-special-member-functions 规则,提示缺失虚析构delete 本身没越界,只是漏掉了派生类清理逻辑
情况可以不加虚析构函数不是所有基类都要加虚析构函数。关键看是否允许“通过基类指针 delete 派生类对象”。
std::string、std::vector 等标准容器类没有虚析构函数,因为它们**不设计为被继承**(C++20 起还加了 final)std::less)、函数对象类,通常按值传递,不涉及多态删除std::shared_ptr 接口)final,编译器可确定无派生类,虚析构失去意义虚析构函数是解决多态删除泄漏的底层机制,但实际工程中应尽量避免裸指针多态删除。
std::shared_ptr 或 std::unique_ptr,它们默认支持虚析构语义(只要 Base::~Base() 是虚的)std::unique_ptr 可安全管理派生类对象:std::unique_ptrp = std::make_unique (); // OK
std::vector 替代 int*),即使析构非虚,资源也会随成员自动释放虚析构函数本身很简单,但它的存在意义常被低估——它不是语法装饰,而是多态对象生命周期管理的契约起点。一旦打破这个契约,泄漏往往静默发生,排查成本远高于预防成本。