当前位置: 首页 > news >正文

【C++】C++的虚析构函数

C++的虚析构函数

  • 1.语法规则:
  • 2.用途:
  • 3.原理:
        • 示例代码:
  • 4. 下面解释为什么基类未定义为析构函数时,析构子类(派生类)对象也能把基类对象析构的原因
    • 4.1核心原理:编译器自动生成的析构函数调用链
      • 4.1.1 对象构造与析构的镜像对称原则
      • 4.1.2 编译器在派生类析构函数中插入隐式代码
    • 4.2 底层机制分步解析
    • 4.3 技术细节说明
      • 4.3.1.this指针调整
      • 4.3.2.继承链处理
      • 4.3.3.与虚函数无关

1.语法规则:

virtual ~析构函数   // 写了虚析构就不能同时再去写普通析构
{}

父类的指针指向子类对象的时候,如果delete释放父类的指针,那么正常情况下只会调用父类的析构函数,不会调用子类的析构函数(释放不彻底)

2.用途:

把父类的析构函数定义成虚析构就能解决
在继承的时候建议把父类的析构函数定义成虚析构

3.原理:

  • 不加virtual,此时采用静态联编(只调用赋值运算左边的类(父类)析构函数)
Cat c1;
Animal *p=c1; // 父类指针指向子类对象
delete p;
  • 加virtual,此时采用动态联编(依据赋值运算右边的类型(子类),先调用子类析构,再调用父类析构)
Cat c1;
Animal *p=c1; // 父类指针指向子类对象
delete p;
示例代码:
#include <iostream>
using namespace std;/*虚析构为了解决特定的问题,发明的一种语法规则。特定的问题:父类的指针指向子类的堆空间,delete父类指针,默认情况下只会调用父类的析构,不会调用子类的析构解决方法:把父类的析构函数定义成虚析构
*/
class Animal
{
public:Animal(){cout<<"父类动物构造"<<endl;}virtual ~Animal(){cout<<"父类动物析构"<<endl;}
};class Cat:public Animal
{
public:Cat(){cout<<"子类猫构造"<<endl;}~Cat(){cout<<"子类猫析构"<<endl;}
};// void showAnimalEat(Animal *other)
// {// other->eat();// delete other;
// }// Cat *c1=new Cat;
// Dog *d1=new Dog;
// showAnimalEat(c1);
// showAnimalEat(d1);int main()
{Cat *c1=new Cat;//父类的指针指向子类对象Animal *p=c1;//释放堆空间// delete c1; delete p;  //加了virtual,析构就正常了,不加virtual析构不彻底/*不加virtual只会调用基类的析构函数而不调用子类的析构函数:父类动物构造子类猫构造父类动物析构*/
}/*
执行结果: 在基类中加了virtual后不管是delete子类指针还是delete父类指针,都会调用子类的析构函数父类动物构造子类猫构造子类猫析构父类动物析构
*/

4. 下面解释为什么基类未定义为析构函数时,析构子类(派生类)对象也能把基类对象析构的原因

\quad 在C++中,当直接通过派生类指针删除对象时,派生类析构函数能够自动调用基类析构函数的原理,主要基于C++语言标准规定的对象销毁顺序编译器生成的隐式代码机制。以下是详细解释:

4.1核心原理:编译器自动生成的析构函数调用链

4.1.1 对象构造与析构的镜像对称原则

  • 构造顺序:基类构造 → 成员构造 → 派生类构造
  • 析构顺序:派生类析构 → 成员析构 → 基类析构(完全相反)

4.1.2 编译器在派生类析构函数中插入隐式代码

\quad 当编译器处理派生类析构函数时,会自动在函数体末尾添加调用基类析构函数的代码。例如:

// 用户编写的派生类析构函数
~Derived() {// 用户定义的清理代码
}// 编译器实际生成的代码
~Derived() {// 1. 用户定义的清理代码// 2. 析构所有成员对象(按声明逆序)// 3. 调用直接基类的析构函数  <-- 关键!
}

4.2 底层机制分步解析

假设有以下类结构:

class Base {
public:~Base() { /* 基类析构 */ }
};class Derived : public Base {Member m;  // 成员对象
public:~Derived() { /* 派生类析构 */ }
};

当执行 delete d(Derived* d)时:

  1. 调用派生类析构函数

在这里插入图片描述
2) 执行用户代码

// 执行用户编写的析构代码

3) 析构成员对象(编译器插入)

// 编译器自动添加成员析构
m.~Member();  // 逆序析构所有成员

4)调用基类析构函数(编译器插入)

// 编译器自动添加基类析构调用
Base::~Base(this);  // 关键步骤!

5)释放内存

operator delete(d);  // 释放整个对象内存

4.3 技术细节说明

4.3.1.this指针调整

  • 当调用基类析构函数时,编译器会自动将this指针调整到基类子对象的起始位置
  • 例如:Base::~Base(this + offset) → 实际是 Base::~Base( (Base*)this )

4.3.2.继承链处理

对于多级继承:

class GrandBase { /*...*/ };
class Base : public GrandBase { /*...*/ };
class Derived : public Base { /*...*/ };

编译器生成的 ~Derived() 会:

~Derived() {// 用户代码// 析构成员Base::~Base(this);    // 调用直接基类// 实际在~Base()中会再调用GrandBase::~GrandBase()
}

4.3.3.与虚函数无关

此过程完全在编译时确定:

  • 不依赖虚函数表(vtable)
  • 不涉及运行时动态查找
  • 是静态绑定的函数调用

相关文章:

  • Linux基本命令篇 —— head命令
  • 什么是P2P 网络(Peer-to-Peer Network)
  • linux运维学习第10周
  • FastAPI+Sqlite+HTML的登录注册与文件上传系统:完整实现指南
  • 命令模式 - Flutter中的操作封装大师,把“动作“变成可管理的对象!
  • 数据同步工具对比:Canal、DataX与Flink CDC
  • stm32hal模块驱动(2)bmi270气压计
  • 数据结构之单链表
  • 爬虫实战之图片及人物信息爬取
  • 华为云Flexus+DeepSeek征文 | 华为云 ModelArts Studio 赋能 AI 法务:合同审查与法律文件生成系统
  • 【硬核数学】4. AI的“寻路”艺术:优化理论如何找到模型的最优解《从零构建机器学习、深度学习到LLM的数学认知》
  • Leetcode 3598. Longest Common Prefix Between Adjacent Strings After Removals
  • 滑块验证码(1)
  • 【blender】使用bpy对一个obj的不同mesh进行不同的材质贴图(涉及对bmesh的操作)
  • ViTMatte:利用预训练的基础视觉Transformer提升图像抠图性能
  • 云计算在布莱克-斯科尔斯模型中的应用:解析解、蒙特卡洛模拟与可视化-AI云计算数值分析和代码验证
  • Node.js特训专栏-实战进阶:11. Redis缓存策略与应用场景
  • 【更新至2024年】1999-2024年各省城镇居民人均消费支出数据(无缺失)
  • 八股文——JAVA基础:String s1 = new String(“abc“);这句话创建了几个字符串对象?
  • window11 本地安装 MySQL8.0