C++入门全集(5):内存管理

前言

一、内存区域划分

二、C++的内存管理方式

2.1 对内置类型

2.2 对自定义类型

三、new和delete的底层实现

四、new和delete的原理

五、定位new

六、malloc/free和new/delete

 

前言

在C++中,内存管理是不可避免的一门必修课。C++对内存的自由度使其获得了更高的性能,以及更高的难度。内存泄漏往往是每个C++学习者绕不开的错误, 而内存管理的水平高低也能看出一个编程者的能力。

在C语言中,我们学习了malloc、calloc、realloc和free,对C语言的内存管理也有了大致的接触。

本文中我们来学习C++中的内存管理


一、内存区域划分

C++中,程序的内存区域从低地址到高地址划分如下:

  • 代码段:存储可执行程序的代码和只读常量
  • 数据段:存储已初始化的全局变量和静态变量
  • 堆:用于程序运行时动态内存分配,从低地址向高地址增长
  • 栈:又叫堆栈,存储非静态局部变量/函数参数和返回值等,从高地址向低地址增长

例如: 

globalVar是全局变量,staticGlobalVar是静态全局变量,存储在数据段中;

staticVar是静态局部变量,存储在数据段中;

localVar、num1、char2、pChar3和ptr1都是局部变量,存储在栈中;

*char2是在栈帧中的空间,存储在栈中;*pChar3是常量字符串abcd的第一个字符,在代码段中;

malloc动态开辟出的空间存储在堆上。


二、C++的内存管理方式

2.1 对内置类型

C++兼容C语言,所以C语言的内存管理方式在C++中可以正常使用

相比C语言使用malloc和free等函数进行内存管理,C++提出了自己的内存管理方式:通过newdelete操作符进行动态内存管理

以上是C语言和C++动态申请一个类型大小的空间、多个类型大小的空间和内存释放的方式

 

实际上,面对内置类型,用malloc和new没有本质的区别,最大的区别在于:new可以初始化

在默认情况下,new和malloc一样不会对内置类型进行初始化,但是我们可以提出需求

注意区分初始化和申请多个元素,一个是圆括号一个是方括号

我们new一个数组时也可以进行初始化,例如:

C++不推荐使用malloc和free,我们最好使用new和delete,并且记得务必匹配使用。

对于内置类型,malloc和new区别不大。new是为了自定义类型而生的。

2.2 对自定义类型

我们在创建自定义类型对象的时候,需要调用析构函数,销毁时需要调用构造函数

而如果我们使用malloc和free的话,是不会调用这两个函数的

使用new来为自定义类型对象申请空间,编译器才会调用构造函数为对象初始化;用delete为自定义类型对象释放空间,才会调用析构函数。


三、new和delete的底层实现

new和delete并不是函数,而是用户进行动态内存申请和释放的操作符

但是其底层还是需要调用函数。

new在底层调用operator new这个函数来申请空间,delete在底层调用operator delete函数来释放空间。operator new和operator delete是系统提供的全局函数。

注意:这两个函数不是new和delete的重载函数!!!

虽然函数名中带operator,但并不是重载函数,具有很强的误导性。

我们来看看这两个函数的实现:

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);
}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;
}

通过上面两个全局函数的实现,可以看出operator new实际上也是通过malloc来申请空间的,如果malloc申请空间成功就直接返回,如果失败则执行用户提供的应对措施,如果用户提供该措施则继续申请空间,否则抛出异常。

operator delete最终也是通过free来释放空间的。就像引用的底层也是用指针的方式实现的。

需要注意,构造函数和析构函数不是通过这两个函数来调用的。 


四、new和delete的原理

(1)new的原理

  • 调用operator new函数申请空间
  • 在申请的空间上执行构造函数,完成对象的构造

(2)delete的原理

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

(3)new T[N]的原理

  • 调用operator new[]函数,而operator new[]函数实际上又会调用operator new函数完成N个T类型对象的空间申请
  • 在申请的空间上执行N次构造函数

(4)delete[]的原理

  • 在空间上执行N次析构函数,完成N个对象的资源清理
  • 调用operator delete[]函数,而operator delete[]函数又会调用operator delete函数来释放空间


五、定位new

定位new表达式用于在已分配的原始内存空间调用构造函数初始化一个对象

大部分情况下,我们直接使用new来给对象分配空间

但是有时候需要进行性能优化,我们会直接从内存池中拿空间,使用malloc开空间

平时我们需要分配空间时从操作系统——堆上开空间,每次有需求就要开一次,但是内存池一次从堆上拿走一个内存块(大块空间),就不需要我们重复的去申请,减少了与堆的交互,提升了效率

如果是自定义的对象,对于malloc出来的空间,则需要使用定位new来显式的调用构造函数进行初始化

使用格式:

构造函数不需要传参时:new(指针)类名

构造函数需要传参时:new(指针)类名(参数)

例如:

或者:

销毁的方式:


六、malloc/free和new/delete

malloc/free和new/delete的共同点在于:都是从堆上申请空间,并且需要用户手动释放

不同点在于:

  1. malloc和free是函数,new和delete是操作符
  2. malloc申请的空间不会初始化,new可以进行初始化
  3. malloc申请空间时需要自行计算要开的空间大小,new只需要空间类型和元素个数
  4. malloc的返回类型为void*,需要强制转换类型,new不需要
  5. malloc申请空间失败时返回空指针,需要判空,new失败时抛出异常,需要捕获
  6. 为自定义类型对象申请空间时,malloc和free不会调用构造函数和析构函数,而new和free会调用

如果文章有误,欢迎在评论区指出

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

DA14531在三星手机手写笔的应用让我打开眼镜

手写笔的功能 这是一款内置蓝牙功能的魔性笔,它是遥控器、是照相、切换摄像头、是暂停或者打开播放列表。乃至更多操作-通过不同的手势隔空操作,或者按下触控按键便可轻松搞定。 手写笔硬件设计 内部结构 采用2.3V可循环充电电池,放入手…

【CSP试题回顾】202212-2-训练计划

CSP-202212-2-训练计划 解题思路 输入和初始化: 首先,代码从输入中获取项目的截止日期和项目数量。然后,它初始化一个项目列表,每个项目都有其依赖项、被依赖的项目集合、完成时间、总完成时间(包括依赖链&#xff09…

@RequestBody

目录 概述 深入细节 案例 RequestBody与前端传过来的json数据的匹配规则 指定模型中的属性对应什么key 用Valid校验RequestBody的参数 根据RequestBody的内容来区分使用哪个资源 概述 RequestBody主要用来接收前端传递给后端的json字符串中的数据(请求体中的数据)而最常…

鸿蒙Harmony应用开发—ArkTS声明式开发(通用属性:多态样式)

设置组件不同状态下的样式。 说明: 从API Version 8开始支持。后续版本如有新增内容,则采用上角标单独标记该内容的起始版本。 从API Version 11开始支持另一种写法attributeModifier,可根据开发者需要动态设置属性。 stateStyles stateStyl…

【深度学习笔记】5_8 网络中的网络NiN

注:本文为《动手学深度学习》开源内容,部分标注了个人理解,仅为个人学习记录,无抄袭搬运意图 5.8 网络中的网络(NiN) 前几节介绍的LeNet、AlexNet和VGG在设计上的共同之处是:先以由卷积层构成的…

STL容器之map和set的补充AVL树

一、AVL树 ​ 不同搜索的对比:1.暴力搜索时间复杂度是O(N);2.二分查找的时间复杂度是O(lgN),但是伴随着有序,插入删除挪动数据的成本极高;3.二叉搜索的时间复杂度是高度次数,极端场景会退化为类似链表时间…

MySQL 主从同步模式

MySQL主从同步是一种数据库复制技术,其中一个MySQL数据库服务器(主服务器)上的更改会被自动地传播到一个或多个其他数据库服务器(从服务器)。这有助于提高系统的可伸缩性、可用性和容错性。以下是设置MySQL主从同步的基…

网络编程的学习

思维导图 多路复用代码练习 select完成TCP并发服务器 #include<myhead.h> #define SER_IP "192.168.125.73" //服务器IP #define SER_PORT 8888 //服务器端口号int main(int argc, const char *argv[]) {//1、创建用于监听的套接字int sfd -1;s…

BUUCTF-MISC1

二维码1 1.打开附件 得到一个二维码 2. 查看 拉到kali用binwalk查看 3.文件分离 发现图片中含有压缩包文件 用binwalk -e 文件名 &#xff0c;进行文件分离 得到一个分离后的文件夹 点开&#xff0c;发现是一个压缩包&#xff0c;压缩包中含有加密文档 4.破解密码 用ARCH…

CSS3新特性

简介 继CSS2之后&#xff0c;CSS3增加了很多新的特性&#xff0c;虽然W3C仍在规范中&#xff0c;但是很多新的CSS3属性已经在很多现代浏览器中得到了支持。 CSS3边框 在CSS3中&#xff0c;可以创建圆角边框&#xff0c;添加边框阴影&#xff0c;设置边框图片&#xff0c;利用…

市场复盘总结 20240305

仅用于记录当天的市场情况&#xff0c;用于统计交易策略的适用情况&#xff0c;以便程序回测 短线核心&#xff1a;不参与任何级别的调整&#xff0c;采用龙空龙模式 一支股票 10%的时候可以操作&#xff0c; 90%的时间适合空仓等待 二进三&#xff1a; 进级率中 25% 最常用的…

栈和队列之队列

1.队列 1.1队列的概念 队列&#xff1a;只允许在一端进行插入数据操作&#xff0c;在另一端进行删除数据操作的特殊线性表&#xff0c;队列具有先进先出 FIFO(First In First Out) 入队列&#xff1a;进行插入操作的一端称为队尾 出队列&#xff1a;进行删除操作的一端称为队…