一起来学习C++的动态内存管理

目录
  • 1.new和delete
  • 2.new和delete在底层是怎么实现的:
    • 2.1new底层的实现: 我们先来new一个test类型的空间。
    • 2.2delete底层的实现: 我们执行delete语句,转到反汇编来
    • 2.3new []底层的实现:
    • 2.4delete []的实现:
  • 3.重载new和delete
  • 4.定位new:
  • 5.内存检测函数:_CrtDumpMemoryLeaks();
  • 总结

1.new和delete

C语言内存管理方式在C++中可以继续使用,但有些地方就无能为力而且使用起来比较麻烦,因此C++又提出了自己的内存管理方式:通过new和delete操作符进行动态内存管理

这里在动态申请和释放时一定要匹配使用

但是我们发现即使我们随意使用程序也没有崩溃,但是这些开辟的空间都是默认类型的

我们接下来试一下开辟自定义类型的空间程序是否会崩溃

我们运行程序,发现直接就崩溃了

那为什么自定义的类型就不可以随意使用释放内存的函数呢?这就要探究malloc,free和new,delete的区别。我们先来看malloc和free

然后再看new和delete

2.new和delete在底层是怎么实现的:

2.1new底层的实现: 我们先来new一个test类型的空间。

然后运行起来我们转到反汇编代码看看

总结new的实现就是调用operate new(size_t)函数,函数内部循坏调用malloc如果申请空间成功就返回,如果申请失败就检测是否有应对措施,如果提供就执行措施,再继续malloc,如果未提供就抛出异常。然后再调用构造函数将申请的内存初始化。

2.2delete底层的实现: 我们执行delete语句,转到反汇编来

进入不知名函数

那么我们来看看operator delete(void *pt)函数是如何如何实现的

总结:delete 的实现就是先调用类中的析构函数,将对象中的数据清空,然后调用operator delete(void *pt)函数,将内存释放掉。

2.3new []底层的实现:

执行完这一系列操作后就会调用n次构造函数,将对象构造成功总结:new []就是调用operate new[]函数,在operate new []中调用operate new函数执行刚刚的一系列操作,然后返回申请的内存。再调用n次构造函数将申请的内存初始化。

2.4delete []的实现:

总结:delete函数就是先调用n次析构函数将申请的内存空间中的数据清空,然后再调用operate delete()函数将内存释放掉。

3.重载new和delete

既然有了new和delete这么好的申请内存的方法,那我们为什么还要重新实现new和delete呢?注意这里的重新实现实现new和delete并不是重新实现new和delete的申请内存的方式,而是有时我们再debug版本下调试时需要一些打印输出一些信息,这里我们需要再用new申请内存的时候要实现输出一些信息。比如说打印文件名,调用函数,调用行数等等。我们重新实现new将我们的文件信息,调用函数,调用行数都打印出来

可以看到虽然将函数都打印出来了但是传参的时候需要传许多参数,所以我们可以用宏替换的方式来解决

但是我们需要这些信息都是在调试的时候才需要,所以我们可以条件编译一下只在debug版本底下执行

我们将delete实现重载,

注意这里如果我们将delete自己实现重载之后再调用delete重载函数是不会调用类中的析构函数的,如果我们的对象中涉及了资源的申请那么就不会释放。

4.定位new:

使用场景:在有些场景下我们可能会申请一块内存空间,但是这块内存空间并没有初始化,当我们想要使用这块内存空间时,想给他初始化,对于类类型的对象我们想要给他初始化时,只能调用构造函数初始化,但是构造函数只能在创建时由编译器自动调用(就像人不可以选择自己的出生时间),那我们这时就要使用定位new来给已经申请号的内存中创建一个对象。

定位new的几种使用方式:

释放空间时

定位new的原理:

5.内存检测函数:_CrtDumpMemoryLeaks();

_CrtDumpMemoryLeaks();函数是window操作系统提供的一个api(应用程序接口)函数,当程序中有内存泄漏时就会在底行输出内存泄漏信息。

我们将申请的空间正确释放,可以看到没有任何的提示。

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注我们的更多内容!

(0)

相关推荐

  • C++的动态内存管理你真的了解吗

    目录 前言 用法上 对内置类型 对自定义类型 new/delete底层原理 重载类的专属operatornew和operatordelete 定位new new/delete与malloc/free区别总结 内存泄漏 总结 前言 想必大家对c语言的动态内存分配并不陌生,忘了的小伙伴也可以看看我的这篇文章C语言动态内存分配 c语言的动态内存分配由于有些地方用起来比较麻烦同时检查错误的机制不适合c++,因此c++引入new/delete操作符进行内存管理,下面我们来深入探讨c++为什么要引入new/

  • c++动态内存管理详解(new/delete)

    目录 前言 用法上 对内置类型 对自定义类型 new/delete底层原理 重载类的专属operatornew和operatordelete 定位new new/delete与malloc/free区别总结 内存泄漏 总结 前言 想必大家对c语言的动态内存分配并不陌生,忘了的小伙伴也可以看看我的这篇文章C语言动态内存分配 c语言的动态内存分配由于有些地方用起来比较麻烦同时检查错误的机制不适合c++,因此c++引入new/delete操作符进行内存管理,下面我们来深入探讨c++为什么要引入new/

  • C++动态内存管理详解

    目录 1.C/C++程序地址空间 2.C语言动态内存管理 (1)malloc (2)calloc (3)realloc (4)free 3.C++动态内存管理 (1)C++为什么要设计一套自己专属的动态内存管理方式? (2)new/delete定义 1)new/delete操作内置类型 2)new/delete操作自定义类型 (3)new/delete的实现原理 4.malloc/free和new/delete的区别 共同点: 不同点: 5.内存泄漏 总结 1.C/C++程序地址空间 计算机物理

  • c++动态内存管理与智能指针的相关知识点

    目录 引言 一.介绍 二.shared_ptr类 make_shared函数 shared_ptr的拷贝和引用 shared_ptr自动销毁所管理的对象… 使用动态生存期的资源的类 应用举例:Blob类 定义Blob类 StrBlob的拷贝.赋值和销毁 三.直接管理内存 使用new分配内存 使用new动态分配和初始化对象 动态分配const对象 内存耗尽 使用delete释放内存 基本介绍 举例 四.shared_ptr和new结合使用 new直接初始化share_ptr 初始化时传入可调用对象

  • 一起来学习C++的动态内存管理

    目录 1.new和delete 2.new和delete在底层是怎么实现的: 2.1new底层的实现: 我们先来new一个test类型的空间. 2.2delete底层的实现: 我们执行delete语句,转到反汇编来 2.3new []底层的实现: 2.4delete []的实现: 3.重载new和delete 4.定位new: 5.内存检测函数:_CrtDumpMemoryLeaks(); 总结 1.new和delete C语言内存管理方式在C++中可以继续使用,但有些地方就无能为力而且使用起来

  • C语言初识动态内存管理malloc calloc realloc free函数

    目录 一.为什么存在动态内存分配 二.动态内存函数的使用 1.malloc函数 (1)malloc的定义 (2)malloc函数的注意事项 (3)malloc函数的使用 2.calloc函数 (1)calloc函数的定义 (2)calloc函数的注意事项 (3)calloc函数的使用 3.realloc函数 (1)realloc函数的定义 (2)realloc函数的注意事项 (3)realloc函数的使用 总结 一.为什么存在动态内存分配 在c语言中我们目前掌握的内存开辟方式有: int val

  • C语言的动态内存管理你了解吗

    目录 C/C++内存分配方式 C++内存管理方式 new和delete的使用 new和delete的骚操作 new和delete的区别 重载new和delete 定位new表达式 内存泄露 总结 C/C++内存分配方式 在学习C语言阶段的时候,创建一个变量,编译器会为它分配一块内存.而创建一个C++对象的时候,编译器会为这个对象分配内存,并且调用合适的构造函数进行初始化. 那么编译器的内存分配方式是怎样的呢? 内存分配可以有以下的几种方式 从静态存储区分配.这样的分配方式在程序开始前就可以为对象

  • 超详细分析C语言动态内存管理问题

    目录 一.为什么存在动态内存的分配 二.动态内存函数的介绍 2.1 malloc和free 2.2 calloc 2.3 realloc 三.常见的动态内存错误 3.1 对NULL指针的解引用操作 3.2 对动态开辟空间的越界访问 3.3 对非动态开辟内存使用free释放 3.4 对同一块动态内存多次释放 3.5 动态开辟内存忘记释放(内存泄漏) 四.几个经典的笔试题 五.C/C++程序的内存开辟 六.柔性数组 6.1 柔性数组的特点 6.2 柔性数组的使用 6.3 柔性数组的优势 上期结束了[

  • C语言深入细致讲解动态内存管理

    目录 为什么存在动态内存管理 动态内存函数的介绍 malloc free calloc realloc 常见的动态内存错误 对NULL指针的解引用操作 对动态开辟空间的越界访问 对非动态开辟内存使用free访问 使用free 释放一块动态开辟内存的一部分 对一块动态内存多次释放 对动态内存开辟忘记释放 柔性数组 小结 为什么存在动态内存管理 我们已经掌握的内存开辟方式有: int val = 20;//在栈空间上开辟四个字节 char arr[10] = { 0 };//在栈空间上开辟10个字节

  • 深入理解C++中的new/delete和malloc/free动态内存管理及区别介绍

    malloc/free和new/delete的区别 malloc/free是C/C++标准库的函数:new/delete是C++操作符. malloc/free只是动态分配内存空间/释放空间:new/delete除了分配空间还会调用构造函数和析构函数进行初始化与清理资源. malloc/free需要手动计算类型大小且返回值类型为void*:new/delete可自动计算类型的大小,返回对应类型的指针. malloc/free管理内存失败会返回0:new/delete等的方式管理内存失败会抛出异常

  • C语言动态内存管理的实现

    目录 1. 摘要 2. 为什么存在动态内存管理 3. 动态内存函数 3.1 malloc 3.2 free 3.3 calloc 3.4 realloc 4. 常见的动态内存错误 5. 几个经典笔试题 参考答案 6. 参考文献 1. 摘要 本文主要详解C语言中的动态内存分配 2. 为什么存在动态内存管理 我们先来看一段变量的声明: double x = 1.000000; char str[] = "abcdef"; 好的,上述变量的声明有何特点呢? 请思考一下,我的朋友. 对,没错,

  • C语言动态内存管理分析总结

    目录 什么是动态内存分配 动态内存函数的介绍 free malloc calloc realloc 动态内存管理中常见的错误 对NULL指针的解引用操作 对动态开辟空间的越界访问 对非动态开辟内存使用free释放 使用free释放一块动态开辟内存的一部分 对同一块动态内存多次释放 动态开辟内存忘记释放(内存泄漏) 一些经典的笔试题 题目1 题目2 题目3 题目4 柔性数组 柔性数组的特点 柔性数组的优势 什么是动态内存分配 我们都知道在C语言中,定义变量的时候,系统就会为这个变量分配内存空间,而

  • C语言动态内存管理介绍

    目录 前言: C 语言为内存的分配和管理提供了几个函数: 1.malloc() 用法 2.calloc() 用法 3.realloc() 与 free() 用法 前言: 简单记录一下,内存管理函数 为什么使用动态内存呢? 简单理解就是可以最大限度调用内存 用多少生成多少,不用时就释放而静止内存不能释放 动态可避免运行大程序导致内存溢出 C 语言为内存的分配和管理提供了几个函数: 头文件:<stdlib.h> 注意:void * 类型表示未确定类型的指针  1.malloc() 用法  分配一块

随机推荐