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

C++delete详解剖析

delete的原理

  1. 在空间上执行析构函数,完成对象中资源的清理工作
  2. 调用operator delete函数释放对象的空间

在C++中,delete 操作符的执行过程涉及两个主要步骤:首先调用对象的析构函数来清理资源,然后通过 operator delete 函数释放分配给对象的内存。这两个步骤的具体行为取决于几个因素,包括对象是否是通过 new 动态分配的、对象是否是数组的一部分、以及类是否定义了虚析构函数等。

1. 调用析构函数

决定调用哪个析构函数的因素:
  • 对象的实际类型:如果通过基类指针指向派生类对象并且该基类的析构函数被声明为虚函数,则会根据对象的实际类型(即派生类)调用相应的析构函数。这是实现多态性的重要部分。(这里会感到疑惑不满足多态三同条件,但是编译器会对所有析构函数进行处理,处理成三同)

  • 非虚析构函数的情况:如果基类的析构函数不是虚函数,那么即使通过基类指针指向派生类对象,也只会调用基类的析构函数,这可能导致派生类的部分没有被正确销毁,从而引发资源泄漏等问题。

    • 示例
      class Base {
      public:virtual ~Base() { std::cout << "Base destructor" << std::endl; }
      };class Derived : public Base {
      public:~Derived() override { std::cout << "Derived destructor" << std::endl; }
      };int main() {Base* ptr = new Derived();delete ptr; // 输出: Derived destructor, Base destructorreturn 0;
      }
      
      在这个例子中,由于 Base 的析构函数是虚函数,当删除 ptr 时,程序首先调用了 Derived 类的析构函数,然后才调用 Base 类的析构函数。

2. 调用 operator delete 函数释放内存

一旦对象的析构函数被调用并完成其工作后,接下来就是释放这块内存。这里涉及到 operator delete 函数的调用:

  • 默认 operator delete:如果没有特别定义 operator delete,则使用全局或类特定的 operator delete 来释放内存。默认情况下,C++标准库提供了一个全局的 operator delete 函数,它负责将内存返回给系统。

    • 示例
      MyClass* obj = new MyClass();
      delete obj; // 默认的 operator delete 被调用来释放内存
      
  • 自定义 operator delete:你可以为你的类重载 operator delete,以便以不同的方式管理内存(例如,使用内存池)。

    • 示例
      class MyClass {
      public:void* operator new(size_t size) {std::cout << "Custom allocation" << std::endl;return ::operator new(size);}void operator delete(void* ptr) noexcept {std::cout << "Custom deallocation" << std::endl;::operator delete(ptr);}
      };int main() {MyClass* obj = new MyClass();delete obj; // 使用自定义的 operator deletereturn 0;
      }
      

数组与单个对象的区别

对于动态分配的数组,你需要使用 delete[] 而不是 deletedelete[] 不仅会调用数组中每个元素的析构函数(如果有的话),还会确保整个数组占用的内存块被正确释放。

  • 注意:不要混淆 deletedelete[]。错误地使用它们会导致未定义的行为。

    • 示例
      MyClass* array = new MyClass[3];
      delete[] array; // 正确做法
      

总结

  • 调用析构函数:由对象的实际类型决定,特别是在使用基类指针指向派生类对象时,若基类析构函数为虚函数,则会根据对象的实际类型调用正确的析构函数;否则,默认只调用基类的析构函数。

  • 释放内存:通过 operator delete 完成,可以是默认的也可以是用户自定义的版本。

理解这些机制有助于更有效地管理和优化内存使用,同时避免常见的内存管理错误如内存泄漏和双重释放等。

在这里插入图片描述

http://www.xdnf.cn/news/6473.html

相关文章:

  • 工具类来生成蓝牙指令
  • Java 序列化(Serialization)
  • 奇妙协同效应,EtherNet IP与PROFINET网关优化半导体生产线
  • Git .gitattributes 文件用途详解
  • Baklib知识中台驱动智能服务新实践
  • ZCC6303x-60V/1.2MHz 高效率升压 LED 恒流驱动替代SY7301
  • 【图片识别工具】批量单据识别批量重命名,批量OCR识别图片文字并重命名,批量改名工具的使用步骤和注意事项
  • Modbus TCP转Profinet网关:数字化工厂异构网络融合的核心枢纽
  • pciutils-3.5.5-win64工具的使用方法
  • Java大师成长计划之第23天:Spring生态与微服务架构之服务发现与注册中心
  • 使用命令行拉取 Git 仓库
  • 数学复习笔记 9
  • 自学嵌入式 day 18 - 数据结构 1
  • 嵌软面试每日一阅----FreeRTOS
  • SpringBoot实现简单的API代理服务器
  • Sumsub 活体检测与人证对比 Java Demo
  • pytorch训练可视化工具---TensorBoard
  • Linux 防火墙 firewalld 实战配置教程!
  • 将.pt文件执行图像比对
  • Java详解RabbitMQ工作模式之发布订阅模式
  • 具备AI功能的银河麒麟桌面操作系统已正式上市
  • 手搓传染病模型(SEI - SEIAR )
  • xp_cmdshell bcp 导出文件
  • 道通龙鱼系列-混合翼无人机:垂直起降+长时续航
  • 嵌入式自学第二十二天(5.15)
  • 02、基础入门-Spring生态圈
  • 云上玩转 Qwen3 系列之三:PAI-LangStudio x Hologres构建ChatBI数据分析Agent应用
  • 机器学习第十三讲:独热编码 → 把“红黄蓝“颜色变成001/010/100的数字格式
  • 数据结构之图的应用场景及其代码
  • MySQL 用户权限管理:从入门到精通