C++动态内存管理 解剖new/delete详细讲解(operator new,operator delete)

2024-04-17 22:04

本文主要是介绍C++动态内存管理 解剖new/delete详细讲解(operator new,operator delete),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述
讨厌抄我作业和不让我抄作业的人
讨厌插队和不让我插队的人
讨厌用我东西和不让我用东西的人
讨厌借我钱和不借给我钱的人
讨厌开车加塞和不让我加塞的人
讨厌内卷和打扰我内卷的人

一、C++中动态内存管理

1.new和delete操作内置类型

2.new和delete操作自定义类型

二、operator new与operator delete函数

三、operator new[]与operator delete[]函数

四、new和delete的实现原理

五、完结撒❀

–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀-正文开始-❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–

一、C++中动态内存管理

我们在C语言中所学习的动态内存函数:malloc,calloc,ralloc这些在C++中也可以继续进行使用,但是使用起来比较麻烦,并且一些地方留有不足,因此C++又提出了自己的内存管理方式:通过new和delete操作符进行动态内存管理。

再强调一次:new和delete是两个操作符

1.new和delete操作内置类型

操作方式如下:

void Test()
{// 动态申请一个int类型的空间int* ptr1 = new int;// 动态申请一个int类型的空间并初始化为10int* ptr2 = new int(10);// 动态申请10个int类型的空间int* ptr3 = new int[10];//delete完成空间的销毁delete ptr1;delete ptr2;//销毁连续多个空间delete[] ptr3;
}

new和delete对动态内存开辟的操作就是这么朴实无华。

在这里插入图片描述这里注意一下初始化开辟的一个空间数据用(),开辟多个空间内存用[ ],初始化开辟的多个空间类型数据用{},操作如下:

int* ptr0 = new int[10] {1, 2, 3, 4, 5};
delete[] ptr0;

注意:申请和释放单个元素的空间,使用new和delete操作符,申请和释放连续的空间,使用new[]和delete[],注意:匹配起来使用。

2.new和delete操作自定义类型

操作如下:

class A
{
public:A(int a = 0): _a(a){cout << "A():" << this << endl;}~A(){cout << "~A():" << this << endl;}
private:int _a;
};
int main()
{// new/delete 和 malloc/free最大区别是 new/delete对于【自定义类型】除了开空间还会调用构造函数和析构函数A* p1 = (A*)malloc(sizeof(A));A* p2 = new A(1);free(p1);delete p2;// 内置类型是几乎是一样的int* p3 = (int*)malloc(sizeof(int)); // Cint* p4 = new int;free(p3);delete p4;A* p5 = (A*)malloc(sizeof(A) * 10);A* p6 = new A[10];free(p5);delete[] p6;return 0;
}

大家可以运行上面代码,研究观察new和delete与malloc和free的区别。

注意:在申请自定义类型的空间时,new会调用构造函数,delete会调用析构函数,而malloc与free不会。

二、operator new与operator delete函数

new和delete是用户进行动态内存申请和释放的操作符operator new和operator delete是系统提供的全局函数new在底层调用operator new全局函数来申请空间,delete在底层通过operator delete全局函数来释放空间。

我们针对代码转到反汇编就可以看到:

在这里插入图片描述
而对于operator new其底层实现是通过malloc函数实现的,operator delete其底层是通过free实现的

见下图:

在这里插入图片描述
在这里插入图片描述这里的_free_dbg表示的就是free,因为free函数是一个宏:

#define   free(p)       _free_dbg(p, _NORMAL_BLOCK)

不仅从汇编我们可以认识operator new函数和operator delete函数,我们还可以看一下其是如何实现的:

/*
operator new:该函数实际通过malloc来申请空间,当malloc申请空间成功时直接返回;申请空间
失败,尝试执行空               间不足应对措施,如果改应对措施用户设置了,则继续申请,否
则抛异常。
*/
void* __CRTDECL operator new(size_t size) _THROW1(_STD bad_alloc)
{// try to allocate size bytesvoid* p;while ((p = malloc(size)) == 0)if (_callnewh(size) == 0){// report no memory// 如果申请内存失败了,这里会抛出bad_alloc 类型异常static const std::bad_alloc nomem;_RAISE(nomem);}return (p);
}/*
operator delete: 该函数最终是通过free来释放空间的
*/
void operator delete(void* pUserData)
{_CrtMemBlockHeader* pHead;RTCCALLBACK(_RTC_Free_hook, (pUserData, 0));if (pUserData == NULL)return;_mlock(_HEAP_LOCK);  /* block other threads */__TRY/* get a pointer to memory block header */pHead = pHdr(pUserData);/* verify block type */_ASSERTE(_BLOCK_TYPE_IS_VALID(pHead->nBlockUse));_free_dbg(pUserData, pHead->nBlockUse);__FINALLY_munlock(_HEAP_LOCK);  /* release other threads */__END_TRY_FINALLYreturn;
}/*
free的实现
*/
#define   free(p)               _free_dbg(p, _NORMAL_BLOCK)

上面operator new函数和operator delete函数的实现代码大家简单看一下,理解即可,看不懂没有关系

重点是通过上述两个全局函数的实现知道,operator new 实际也是通过malloc来申请空间,如果malloc申请空间成功就直接返回,否则执行用户提供的空间不足应对措施,如果用户提供该措施就继续申请,否则就抛异常。operator delete 最终是通过free来释放空间的

三、operator new[]与operator delete[]函数

我们在开辟多个类型数据时用new T[N],其内部调用的是operator new[]函数,而operator new[]函数其实就是由operator new函数实现完成的,见下面汇编代码:

在这里插入图片描述

在这里插入图片描述
同理delete[]函数内部实现是由operator delete[]函数实现的,而operator delete[]函数内部实现也是由operator delete函数所实现的:

在这里插入图片描述
在这里插入图片描述

operator new[]函数内部实现逻辑由operator new函数实现,那么为什么还需要operator new[]函数呢?

在上面的学习中,我们强调过new和delete配对使用,而new[]要和delete[]进行配对使用,new[]和delete[]是在创建多个类型数据时进行使用,这里对于内置类型来说的话其实没多大意义(因为内置类型不需要调用构造函数和析构函数),主要在自定义类型上面

new和delete在创建和销毁自定义类型分别会自动调用构造函数和析构函数,所以我们在创建多组自定义类型数据时需要根据自定义类型的个数来确定一共需要调用几次构造函数和析构函数

class A
{
public:A(int a = 0):_a(a){cout << "A(int a = 0)" << this << endl;}~A(){cout << "~A()" << this << endl;}
private:int _a;
};int main()
{A* ptr1 = new A; //operator new + 1次构造A* ptr2 = new A[10];//operator new[] + 10次构造delete ptr1;//1次析构 + operator deletedelete[] ptr2;//10次析构 + operator deletereturn 0;
}

而调用operator new[]函数就实现了对类型数据个数的存储,我们可以看下代码:

class A
{
public:A(int a = 0):_a(a){cout << "A(int a = 0)" << this << endl;}~A(){cout << "~A()" << this << endl;}
private:int _a;
};int main()
{A* ptr1 = new A[3];delete[] ptr1;return 0;
}

按照正常计算的话,new开辟A[3]其大小应为12字节,但是我们转到汇编代码:

在这里插入图片描述
会发现一共是16个字节,这里多出来的4个字节就是用来存储总数据个数所用的。

同理,对于operator delete[]函数也一样,operator new[]函数记录了所创建的总数居的多少,也是为下面operator delete[]函数调用多少次析构函数做铺垫。

对于operator delete[]函数,最后也会在内存空间中会向前偏移4个字节的位置再进行free,将所存储数据个数的4个字节一同free掉。

四、new和delete的实现原理

1.内置类型

如果申请的是内置类型的空间,new和malloc,delete和free基本类似,不同的地方是:new/delete申请和释放的是单个元素的空间,new[]和delete[]申请的是连续空间,而且new在申请空间失败时会抛异常,malloc会返回NULL

2.自定义类型

· new的原理

1. 调用operator new函数申请空间

2. 在申请的空间上执行构造函数,完成对象的构造

· delete的原理

1. 在空间上执行析构函数,完成对象中资源的清理工作

2. 调用operator delete函数释放对象的空间

· new T[N]的原理

1. 调用operator new[]函数,在operator new[]中实际调用operator new函数完成N个对象空间的申请

2. 在申请的空间上执行N次构造函数

· delete[N]的原理

1. 在释放的对象空间上执行N次析构函数,完成N个对象中资源的清理

2. 调用operator delete[]释放空间,实际在operator delete[]中调用operator delete来释
放空间

通过上面的学习我们可以发现,对于操作符,其内部都是由一条一条指令进行实现的,所以我们使用操作符 ,在编译器编译时就会将操作符转换成每一条指令执行。

五、完结撒❀

如果以上内容对你有帮助不妨点赞支持一下,以后还会分享更多编程知识,我们一起进步。
最后我想讲的是,据说点赞的都能找到漂亮女朋友❤
在这里插入图片描述

这篇关于C++动态内存管理 解剖new/delete详细讲解(operator new,operator delete)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/912976

相关文章

Python中的魔术方法__new__详解

《Python中的魔术方法__new__详解》:本文主要介绍Python中的魔术方法__new__的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、核心意义与机制1.1 构造过程原理1.2 与 __init__ 对比二、核心功能解析2.1 核心能力2.2

使用Node.js制作图片上传服务的详细教程

《使用Node.js制作图片上传服务的详细教程》在现代Web应用开发中,图片上传是一项常见且重要的功能,借助Node.js强大的生态系统,我们可以轻松搭建高效的图片上传服务,本文将深入探讨如何使用No... 目录准备工作搭建 Express 服务器配置 multer 进行图片上传处理图片上传请求完整代码示例

C++ vector的常见用法超详细讲解

《C++vector的常见用法超详细讲解》:本文主要介绍C++vector的常见用法,包括C++中vector容器的定义、初始化方法、访问元素、常用函数及其时间复杂度,通过代码介绍的非常详细,... 目录1、vector的定义2、vector常用初始化方法1、使编程用花括号直接赋值2、使用圆括号赋值3、ve

如何高效移除C++关联容器中的元素

《如何高效移除C++关联容器中的元素》关联容器和顺序容器有着很大不同,关联容器中的元素是按照关键字来保存和访问的,而顺序容器中的元素是按它们在容器中的位置来顺序保存和访问的,本文介绍了如何高效移除C+... 目录一、简介二、移除给定位置的元素三、移除与特定键值等价的元素四、移除满足特android定条件的元

Python获取C++中返回的char*字段的两种思路

《Python获取C++中返回的char*字段的两种思路》有时候需要获取C++函数中返回来的不定长的char*字符串,本文小编为大家找到了两种解决问题的思路,感兴趣的小伙伴可以跟随小编一起学习一下... 有时候需要获取C++函数中返回来的不定长的char*字符串,目前我找到两种解决问题的思路,具体实现如下:

C++ Sort函数使用场景分析

《C++Sort函数使用场景分析》sort函数是algorithm库下的一个函数,sort函数是不稳定的,即大小相同的元素在排序后相对顺序可能发生改变,如果某些场景需要保持相同元素间的相对顺序,可使... 目录C++ Sort函数详解一、sort函数调用的两种方式二、sort函数使用场景三、sort函数排序

python连接本地SQL server详细图文教程

《python连接本地SQLserver详细图文教程》在数据分析领域,经常需要从数据库中获取数据进行分析和处理,下面:本文主要介绍python连接本地SQLserver的相关资料,文中通过代码... 目录一.设置本地账号1.新建用户2.开启双重验证3,开启TCP/IP本地服务二js.python连接实例1.

基于Python和MoviePy实现照片管理和视频合成工具

《基于Python和MoviePy实现照片管理和视频合成工具》在这篇博客中,我们将详细剖析一个基于Python的图形界面应用程序,该程序使用wxPython构建用户界面,并结合MoviePy、Pill... 目录引言项目概述代码结构分析1. 导入和依赖2. 主类:PhotoManager初始化方法:__in

Nginx中配置HTTP/2协议的详细指南

《Nginx中配置HTTP/2协议的详细指南》HTTP/2是HTTP协议的下一代版本,旨在提高性能、减少延迟并优化现代网络环境中的通信效率,本文将为大家介绍Nginx配置HTTP/2协议想详细步骤,需... 目录一、HTTP/2 协议概述1.HTTP/22. HTTP/2 的核心特性3. HTTP/2 的优

Java图片压缩三种高效压缩方案详细解析

《Java图片压缩三种高效压缩方案详细解析》图片压缩通常涉及减少图片的尺寸缩放、调整图片的质量(针对JPEG、PNG等)、使用特定的算法来减少图片的数据量等,:本文主要介绍Java图片压缩三种高效... 目录一、基于OpenCV的智能尺寸压缩技术亮点:适用场景:二、JPEG质量参数压缩关键技术:压缩效果对比