把类的析构函数写成虚函数的用意

本文通过一个C++示例代码展示了当基类的析构函数未声明为虚函数时,如何导致内存泄漏的问题,并解释了使用虚析构函数的重要性。

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

#include <iostream.h>
class Base 
public: 
Base() { mPtr = new int; } 
~Base() { delete mPtr; cout<<"Base::Destruction"<<endl;} 
private: 
  int* mPtr; 
} ;

class Derived : public Base 
public: 
  Derived() { mDerived = new long; } 
  ~Derived() { delete mDerived; cout<<"Derived::Destruction"<<endl;} 
private: 
  long* mDerived; 
} ;

void main() 
  Base* p = new Derived; 
  delete p; 
输出结果只有:Base::Destruction
以上代码会产生内存泄露,因为new出来的是Derived类资源,采用一个基类的指针来接收,析构的时候,编译器因为只是知道这个指针是基类的,所以只将基类部分的内存析构了,而不会析构子类的,就造成了内存泄露,如果将基类的析构函数改成虚函数,就可以避免这种情况,因为虚函数是后绑定,其实就是在虚函数列表中,析构函数将基类的析构函数用实际对象的一组析构函数替换掉了,也就是先执行子类的虚函数再执行父类的虚函数,这样子类的内存析构了,父类的内存也释放了,就不会产生内存泄露。
注:
1.析构函数其实是一个函数,不论子类还是父类,虽然可能看起来名字不一样。而且析构函数执行过程都是执行子类再到父类。
2.多态的时候一定要将析构函数写成虚函数,防止内存泄露,各个子类维护自己内部数据释放。

virtual 是实现多态的基础
它使得具体的函数跳转从编译时推迟到运行时然而构造函数的调用是编译器期间就决定的,因此它不能为虚
### C++ 中虚析构函数作用及必要性 在 C++ 的面向对象编程中,虚析构函数的主要作用是在多态环境中确保基指针指向的派生对象能够正确销毁并释放资源。以下是关于虚析构函数的具体分析: #### 1. **虚析构函数的核心功能** 当使用基指针管理派生对象时,如果没有定义虚析构函数,则仅会调用基析构函数,而不会调用派生析构函数[^3]。这种行为可能导致派生中的资源未被正确释放,从而引发内存泄漏或其他潜在问题。 为了防止这种情况发生,可以将基析构函数声明为 `virtual` 型。这样一来,即使通过基指针删除派生对象,也能按照正确的顺序依次调用派生和基析构函数[^2]。 --- #### 2. **虚析构函数的重要性** 虚析构函数的存在对于实现安全的对象销毁至关重要,尤其是在涉及动态分配资源的情况下。具体来说: - 如果基析构函数不是虚函数,那么通过基指针删除派生对象时,只会调用基析构函数,而忽略派生的部分[^4]。 - 这种情况可能会导致派生中某些资源(如动态分配的内存、文件句柄等)无法得到适当处理,进而引起程序错误或性能下降。 因此,在设计支持多态的基时,通常建议将其析构函数声明为虚函数,以便在任何情况下都能保证完整的对象清理流程[^5]。 --- #### 3. **代码示例** 以下是一个简单的例子,展示了虚析构函数的实际应用效果: ```cpp #include <iostream> using namespace std; class Base { public: virtual ~Base() { cout << "Base destructor called." << endl; } }; class Derived : public Base { public: ~Derived() { cout << "Derived destructor called." << endl; } }; int main() { Base* obj = new Derived(); delete obj; return 0; } ``` 运行上述代码的结果如下: ``` Derived destructor called. Base destructor called. ``` 可以看到,由于 `Base` 析构函数被标记为虚拟 (`virtual`),所以在通过基指针删除派生对象时,不仅调用了派生析构函数,还随后调用了基析构函数[^1]。 --- #### 4. **注意事项** 尽管虚析构函数提供了重要的安全保障,但也需要注意其带来的额外开销。每当一个拥有至少一个虚函数时,编译器会在该内部创建一张虚函数表 (vtable),用于存储虚函数的相关信息。这意味着每个实例化出来的对象都会携带一定的附加数据结构,可能稍微增加内存消耗。 因此,只有那些真正需要支持多态语义的基才应该为其析构函数设置为虚函数;而对于其他普通用途的而言,保持普通的非虚析构函数即可满足需求。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值