C++中析构函数不设为virtual导致内存泄漏示例

一、问题示例

cpp 复制代码
#include <iostream>
using namespace std;

class Base {
public:
    Base() { cout << "Base constructor\n"; }
    ~Base() { cout << "Base destructor\n"; } // 不是 virtual
};

class Derived : public Base {
public:
    Derived() { cout << "Derived constructor\n"; }
    ~Derived() { cout << "Derived destructor\n"; }
};

int main() {
    Base* ptr = new Derived();  // 用 Base 指针指向 Derived 对象
    delete ptr;                 // 只调用 Base 的析构函数
    return 0;
}

二、输出结果


注意:没有调用 Derived 的析构函数

三、问题解释

为什么会这样?

  • Base 的析构函数不是 virtual

  • 当你执行 delete ptr,编译器只看 ptr 的静态类型(即 Base*),它就只调用 ~Base()

  • 因此,不会调用 Derived 的析构函数,可能会导致:

    • Derived 析构中释放的资源没有释放(内存泄漏);
    • 或者 Derived 析构中有清理逻辑没执行(行为错误);

四、修复方法

Base 的析构函数设为 virtual

cpp 复制代码
class Base {
public:
    virtual ~Base() { cout << "Base destructor\n"; }  // virtual 析构
};

正确输出:


五、图示原理(简化)

如果你用 virtual

复制代码
Base vtable:
+----------------+
|  &Base::~Base  |  ← 原始虚函数地址
+----------------+

Derived vtable:
+--------------------+
| &Derived::~Derived | ← 覆盖虚析构函数
+--------------------+

delete ptr 会通过 vtable 跳转到 Derived::~Derived → 再调用 Base::~Base

六、总结

是否 virtual delete 派生类指针时行为 是否内存泄漏
只调用基类析构函数 有可能
调用派生 → 再基类析构 安全

七、经验建议

只要类中有虚函数(即打算支持多态)或你"可能"通过基类指针删除派生类对象,就应该将析构函数声明为 virtual

最好这样写:

cpp 复制代码
class Base {
public:
    virtual ~Base() = default;
};
相关推荐
fs哆哆13 分钟前
在VB.net中一维数组,与VBA有什么区别
java·开发语言·数据结构·算法·.net
johnZhangqi18 分钟前
深圳大学-计算机信息管理课程实验 C++ 自考模拟题
java·开发语言·c++
Sally璐璐32 分钟前
Go语言变量声明与初始化详解
java·开发语言·golang
luofeiju1 小时前
交叉编译笔记
开发语言
StudyWinter1 小时前
【C++】仿函数和回调函数
开发语言·c++·回调函数·仿函数
C4程序员2 小时前
北京JAVA基础面试30天打卡14
java·开发语言·面试
黑客影儿2 小时前
Go特有的安全漏洞及渗透测试利用方法(通俗易懂)
开发语言·后端·安全·web安全·网络安全·golang·系统安全
君鼎3 小时前
Effective C++ 条款55:熟悉Boost库
c++
你好,我叫C小白3 小时前
C语言 常量,数据类型
c语言·开发语言·数据类型·常量
小红帽2.03 小时前
从ioutil到os:Golang在线客服聊天系统文件读取的迁移实践
服务器·开发语言·golang