【C++】动态内存管理(79分钟写的文章哪里看不懂了,快来学)

news2025/2/24 14:43:23

动态内存管理目录:

一、C/C++内存分布

 在学习了C/C++内存区域的划分后,我们来做几道题巩固一下:

1. 选择题:选项 : A.栈  B.堆  C.数据段(静态区)  D.代码段(常量区)globalVar在哪里?____  staticGlobalVar在哪里?____staticVar在哪里?____  localVar在哪里?____num1 在哪里?____char2在哪里?____ *char2在哪里?___pChar3在哪里?____ *pChar3在哪里?____ptr1在哪里?____ *ptr1在哪里?____

2. 填空题:sizeof(num1) = ____;sizeof(char2) = ____;    strlen(char2) = ____;sizeof(pChar3) = ____;   strlen(pChar3) = ____;sizeof(ptr1) = ____;

二、C语言动态内存管理方式

malloc:

calloc:

realloc:

free:

 面试题:malloc/calloc/realloc的区别?

三、C++动态内存管理方式(operator new/delete+构造/析构)

3.1new/delete 操作内置类型

3.2new/delete 操作自定义类型

四、 operator new 与 operator delete(探究new操作符底层)

五、定位new表达式(了解)

六、常见面试题

1、malloc/free 和 new/delete 的区别(从用法功能和底层去理解)

2、内存泄漏(不是空间丢了)

什么是内存泄漏

内存泄漏的危害

如何避免内存泄漏


一、C/C++内存分布

在C语言阶段,我们常说局部变量存储在栈区,动态内存中的数据存储在堆区,静态变量存储在静态区,常量和全局变量存储在常量区,其实这里我们所说的栈区、堆区、静态区以及常量区都是 虚拟进程地址空间 的一部分,其中具体内存区域的划分如下:

这个图强烈建议,啃啃啃啃啃啃啃啃啃   

  •  栈:又叫堆栈,用于存储非静态局部变量、函数参数以及函数返回值等等,栈是向下增长的(栈帧就像是一次性水杯)
  • 堆:用于程序运行时进行动态内存分配,堆是向上增长的
  • 数据段 (静态区):Linux 中通常叫作数据段,用于存储存储全局数据和静态数据(静态区不只是有静态变量)
  • 代码段 (常量区):Linux 中通常叫作代码段,用于存储可执行的代码指令和只读常量

 在学习了C/C++内存区域的划分后,我们来做几道题巩固一下:

int globalVar = 1;
static int staticGlobalVar = 1;
void Test()
{
	static int staticVar = 1;
	int localVar = 1;
	int num1[10] = { 1, 2, 3, 4 };
	char char2[] = "abcd";
	const char* pChar3 = "abcd";
	int* ptr1 = (int*)malloc(sizeof(int) * 4);
	int* ptr2 = (int*)calloc(4, sizeof(int));
	int* ptr3 = (int*)realloc(ptr2, sizeof(int) * 4);
	free(ptr1);
	free(ptr3);
}

1. 选择题:
选项 : A.栈  B.堆  C.数据段(静态区)  D.代码段(常量区)
globalVar在哪里?____  staticGlobalVar在哪里?____
staticVar在哪里?____  localVar在哪里?____
num1 在哪里?____
char2在哪里?____ *char2在哪里?___
pChar3在哪里?____ *pChar3在哪里?____
ptr1在哪里?____ *ptr1在哪里?____

分析:上半区比较简单,前面加static和全局的都在静态区,在函数中开辟的且没有static的都是栈

下半区是难点,请好好观察我下面画的图:

可以看出,char2 pChar3 ptr1由于都在Test中,所以他们地址的空间开在了栈帧,但是他们的内容在哪里,取决于类型和开辟方式:对于char2来说,只是普通开辟,所以对象也在栈;对于pChar3来说,类型前有const,它是一个指针,指向代码段的 “abcd”,所以 *pchar3 在代码段;对于*ptr1来说,开辟方式是在堆区开辟,所以*ptr1的数据就是在堆区

2. 填空题:
sizeof(num1) = ____;
sizeof(char2) = ____;    strlen(char2) = ____;
sizeof(pChar3) = ____;   strlen(pChar3) = ____;
sizeof(ptr1) = ____;

sizeof是操作符/关键字,后面可以不加括号,而直接跟类型

另外,sizeof计算的是变量所占空间的字节数

而strlen是函数,调用必须加括号,且strlen计算的是字符串中字符的个数(不包括'\0')

 对于sizeof指针而言,指针就是地址,所以32位和64位的大小不一样,所以是4/8


二、C语言动态内存管理方式

在C语言中我们使用 malloc/calloc/realloc/free 函数来进行动态内存管理:

malloc:

int* p1 = (int*)malloc(sizeof(int));
	if (p1 == NULL)
	{
		perror("malloc fail");
		exit(-1);
	}

 这里检查空是因为编译器不严谨,所以得加上判断,而且开辟失败的时候,malloc返回的是空指针,所以可以这样检查

calloc:

int* p2 = (int*)calloc(4, sizeof(int));
	if (p2 == NULL)
	{
		perror("calloc fail");
		exit(-1);
	}

realloc:

int* p3 = (int*)realloc(p2, sizeof(int) * 10);
	if (p3 == NULL)
	{
		perror("realloc fail");
		exit(-1);
	}

free:

 

free(p1);
free(p3);

 面试题:malloc/calloc/realloc的区别?

  • malloc 用于开辟一块动态内存,使用时需要指定开辟的空间大小 (字节),如果开辟成功返回空间的起始地址,如果开辟失败返回 NULL且不会初始化(所以new就出现了)
  • calloc 的用法和 malloc 类似,只是它有两个参数,第一个参数为元素个数,第二个参数为每个元素的大小,并且它会将该空间中的数据全部初始化为0
  • realloc 用于空间的扩容/缩容,它有两个参数,第一个参数为需要调整的动态内存的起始地址,第二个参数为调整后的空间大小,如果第一个参数为 NULL,则它等价于 malloc;如果扩容,编译器会检查原空间后是否有足够的空间,如果足够,就直接扩容并返回原空间的起始地址,如果不够,就新开辟一块空间,然后将原空间的数据拷贝到新空间并返回新空间的地址,最后再释放原空间;如果缩容,编译器会直接新开辟一块空间,然后拷贝原空间数据到新空间并返回新空间的地址,再释放原空间。

三、C++动态内存管理方式(operator new/delete+构造/析构)

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

3.1new/delete 操作内置类型

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

如果你忘了什么是内置类型,那么请看从构造函数开始看

对于内置类型,C语言和C++内存管理方式没有明显区别,只是C++中使用 new 操作符来替代C语言中的 malloc/calloc 函数,使用 delete 操作符来替代 free 函数 ;

同时,由于 new 和 delete 是操作符/关键字,而不是函数,所以它们后面不需要跟括号,而是直接跟类型即可;另外,new 可以在开辟空间的同时进行初始化(在构造函数的基础上)

注:C++不支持扩容,要扩容都是自己开辟新空间、拷贝数据,然后再销毁原空间

void Test()
{
	//申请单个空间不初始化
	int* p1 = new int;

	//申请单个空间并初始化
	int* p2 = new int(10);

	//申请连续空间不初始化
	int* p3 = new int[10];

	//申请连续空间并初始化
	int* p4 = new int[10]{ 1,2,3,4,5 };

	//释放单个空间
	delete p1;
	delete p2;

	//释放多个空间
	delete[] p3;
	delete[] p4;
}

所以管理对象和管理对象数组还是有所差异滴:

 

 申请和释放单个元素的空间,使用 new 和 delete 操作符,申请和释放连续的空间,使用 new[] 和 delete[],注意二者一定要匹配使用,即不能用 delete 来释放 new[] 开辟的空间

3.2new/delete 操作自定义类型

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次析构函数,完成N个对象中资源的清理;
  • 调用 operator delete[] 释放空间,实际在 operator delete[] 中调用operator delete来释放空间;

C++动态内存管理和C语言动态内存管理最大的不同在于二者对自定义类型的处理:C语言 malloc/calloc/realloc 函数只负责开辟空间,free 函数只负责销毁空间;而C++在申请自定义类型的空间时,new 会调用构造函数,delete 会调用析构函数


class A
{
public:
A (int a=0 ):_a(a)
{
	cout << "A Construct" << this << endl;
}
~A()
{
	cout << "~A()"<< this << endl;
}
private:
int _a;
};


四、 operator new 与 operator delete(探究new操作符底层)

在C++中,new 和 delete 是用户进行动态内存申请和释放的操作符,operator new 和 operator delete 是系统提供的全局函数new 在底层调用 operator new 全局函数来申请空间,delete 在底层通过 operator delete 全局函数来释放空间。(所以他们之间是调用关系,不是重载关系)

需要特别注意的是,operator new 和 operator delete 函数不是运算符重载,因为它们的参数没有自定义类型,而是库里面实现的全局函数,仅仅是将它们取名为 operator 而已,很多C++的初学者都会被二者的函数名所误导。

C++底层的 operator new 和 operator delete 函数如下:

 

我们可以通过查看反汇编代码来验证 new 和 delete 的底层调用:

 而对于new[] 和 delete[] 来说,它们通过调用 operator new[] 和 operator delete[] 函数来实现其功能,但是其实 operator new[] 和 operator delete[] 底层也是调用的 operator new 和 operator delete 函数:

 通过上述的实验我们知道 operator new 实际也是通过 malloc 来申请空间,如果 malloc 申请空间成功就直接返回,否则执行用户提供的空间不足应对措施,如果用户提供了该措施就继续申请,否则就抛异常operator delete 最终是通过 free 来释放空间的 

所以,new其实是封装了malloc,申请内存失败,就会bad allocation 这样才更符合C++面向对象处理问题的机制


五、定位new表达式(了解)

定位 new 也叫 replacement new,定位 new 表达式是在已分配的原始内存空间中调用构造函数初始化一个对象;其使用格式如下:

new(place_address) type 或者 new (place_address) type(initializer-list)

 使用场景

定位 new 表达式在实际中一般是配合内存池使用 – 因为内存池分配出的内存没有初始化,所以如果是自定义类型的对象,需要使用 new 的定义表达式进行显示调构造函数进行初始化;而内存池在后面我们会详细学习,此处我们了解一下即可。

简单理解一下内存池:

假设半山腰有一个村子,但由于各种原因村子中没有水喝,所以人们每次喝水都只能到山下的公共水井处排队打水,但是呢排队很慢,所以村长就用抽水机+水管联通水井在自己家建了一个蓄水池,以后要用水就直接到蓄水池中去取即可,而不用再到山下去排队打水了,大大提高了效率

上述例子中全村公用的水井就相当于堆,其他村民排队打水就相当于 malloc/calloc/realloc 函数向堆区申请空间,而村长家的蓄水池就相当于我们的主角 – 内存池,内存池的建立可以使得我们申请空间的效率变得很高


六、常见面试题

1、malloc/free 和 new/delete 的区别(从用法功能和底层去理解)

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

  • malloc和free是函数,new和delete是操作符
  • 申请内置类型空间时,malloc申请的空间不可以初始化,new可以初始化(对于自定义类型)
  • 申请自定义类型对象时,malloc/free只会开辟空间,不会调用构造函数与析构函数,而new在申请空间后会调用构造函数完成对象的初始化,delete在释放空间前会调用析构函数完成空间中资源的清理;
  • malloc申请空间时,需要手动计算空间大小并传递,new只需在其后跟上空间的类型即可,如果是多个对象,[]中指定对象个数即可;
  • malloc的返回值为void*, 在使用时必须强转,new不需要,因为new后跟的是空间的类型
  • malloc申请空间失败时,返回的是NULL,因此使用时必须判空,new不需要,但是new需要捕获异常

2、内存泄漏(不是空间丢了)

什么是内存泄漏

通俗易懂的话来讲:就是占着茅坑布莱斯,我虽然这块空间不用了,但我就是不释放,是指针丢了,找不到了,而不是内存丢了

定义:内存泄漏指因为疏忽或错误造成程序未能释放已经不再使用的内存的情况;内存泄漏并不是指内存在物理上的消失,而是应用程序分配某段内存后,因为设计错误,失去了对该段内存的控制,因而造成了内存的浪费

内存泄漏的危害

在小程序小代码中,内存泄漏的危害几乎为0,但是对于大公司,比如王者荣耀,当发生内存泄露而且是慢性不好检测出来的时候,由于空间一直被占,回收不回来,这就会导致服务器挂掉等问题

定义:短期运行的程序发生内存泄露危害不大,因为当程序结束时动态申请的空间全部都会被回收;长期运行的程序出现内存泄漏,影响很大,如操作系统、后台服务等等,出现内存泄漏会导致响应越来越慢,最终卡死

如何避免内存泄漏

  1. 工程前期良好的设计规范,养成良好的编码规范,申请的内存空间记着匹配的去释放;(注:这个是理想状态,但是如果碰上异常时,就算注意释放了,也还是可能会出问题,需要下一条智能指针来管理才有保证)
  2. 采用RAII思想或者智能指针来管理资源;
  3. 有些公司内部规范使用内部实现的私有内存管理库;这套库自带内存泄漏检测的功能选项;
  4. 出问题了使用内存泄漏工具检测。(注:很多工具都不够靠谱,或者收费昂贵)

总结:内存泄漏非常常见,解决方案分为两种:

 1、事前预防型;如智能指针等。2、事后查错型;如泄漏检测工具


希望这篇文章可以给你带来收获!!

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【2023知乎评论爬虫】我用Python爬虫爬了2386条知乎评论!

文章目录 一、爬取目标二、展示爬取结果三、爬虫代码讲解3.1 分析知乎页面3.2 爬虫代码 四、同步视频五、完整源码 您好&#xff0c;我是 马哥python说&#xff0c;一枚10年程序猿。 一、爬取目标 前些天我分享过一篇微博的爬虫&#xff1a;https://blog.csdn.net/solo_msk/a…

Spring 的注入

目录 一、注入&#xff08;Injection&#xff09; 1、什么是注入 &#xff08;1&#xff09;为什么需要注入 &#xff08;2&#xff09;如何进行注入 2、Spring 注入原理分析&#xff08;简易版&#xff09; 二、Set 注入详解 1、JDK 内置类型 &#xff08;1&#xff09…

【修复版】2023新版塔罗 算八字测运易理风水 取名 源码平台 搭建教程

全线修复&#xff0c;欢迎你拿其他家的来比&#xff0c;叫他们发测试连接去测试对比&#xff0c;眼睛骗不了人 很多说自家的是修复版&#xff0c;能像我这样把修复的列出来嘛&#xff0c;顾客朋友也可以直接去问他&#xff1a;你到底修复了个啥&#xff1f;&#xff1f;&#…

深入探索智能问答:从检索到生成的技术之旅

目录 一、智能问答概述1. **语义理解**2. **知识库和数据库**3. **上下文感知**4. **动态学习和自适应** 二、发展历程1. **基于规则的系统**2. **统计方法的兴起**3. **深度学习和神经网络的突破**4. **预训练模型** 三、智能问答系统的主要类型四、基于知识库的问答系统五、基…

如何使用记事本制作一个简陋的小网页(1)

1、创建一个记事本 2、将记事本的尾缀进行修改&#xff0c;变为html html本质上是一种标签并不是一种语言&#xff0c;其最主要的功能就是对页面设置和页面的文本样式进行修改和修缮。 3、右键被修改了尾缀的文本&#xff0c;选择使用记事本的打开方式。 打开完毕后&#xf…

什么是JavaScript的事件驱动编程(event-driven programming)?

聚沙成塔每天进步一点点 ⭐ 专栏简介⭐ 事件&#xff08;Event&#xff09;⭐ 事件监听器&#xff08;Event Listener&#xff09;⭐ 回调函数&#xff08;Callback Function&#xff09;⭐ 非阻塞和异步⭐ 事件循环&#xff08;Event Loop&#xff09;⭐ 触发事件&#xff08;…

「C++程序设计 (面向对象进阶)」学习笔记・二

0、引言 本专栏的系列文章是在学习 北京邮电大学 崔毅东 老师的《C程序设计 (面向对象进阶)》课程过程中整理的。欢迎前往专栏了解更多相关内容~ &#x1f600; 有关于现代 C 的基本介绍&#xff0c;请前往《现代C基本介绍》&#xff01; &#x1f514; 先决条件 本专栏的系列…

【测试开发】答疑篇 · 什么是软件测试

【测试开发】答疑篇 文章目录 【测试开发】答疑篇1. 生活中的测试2. 什么是软件测试3. 为什么要有测试/没有测试行不行4. 软件测试和软件开发的区别5. 软件测试和软件调试之间的区别6. 软件测试的岗位7. 优秀测试人员具备的素质 【测试开发】答疑篇 软件不一定是桌面应用&#…

公众号迁移线上公证如何办?

公众号账号迁移的作用是什么&#xff1f;只能变更主体吗&#xff1f;微信公众平台的帐号迁移功能可将原公众号的粉丝、文章素材、违规记录、留言功能、名称等迁移至新的公众号。通过迁移可以实现公众号的公司主体变更、粉丝转移、开通留言功能、服务号转为订阅号等作用。因此不…

Qt基于paintEvent自定义CharView

Qt基于paintEvent自定义CharView 鼠标拖动&#xff0c;缩放&#xff0c;区域缩放&#xff0c; 针对x轴&#xff0c;直接上代码 charview.h #ifndef CHARVIEW_H #define CHARVIEW_H#include <QWidget> #include <QPainter> #include <QPaintEvent> #inclu…

Kotlin使用infix关键字构建高可读性代码

自从Google推出Kotlin作为官方开发语言后&#xff0c;一堆像我这样的习惯Java开发的程序员从不习惯到爱上这门语言只用了很短的时间&#xff0c;相比于Java语言&#xff0c;kotlin集各家所长&#xff0c;可以使用更少的代码&#xff0c;实现更多更复杂的需求&#xff0c;而且可…

腾讯mini项目-【指标监控服务重构】2023-08-18

今日已办 watermill 将 key 设置到 message 中 修改 watermill-kafka 源码 将 key 设置到 message.metadata中 接入 otel-sdk 添加 middleware resolveUpstreamCtx 解析上游上下文&#xff0c;开启根Span添加 middleware middleware.InstantAck - 马上ACK&#xff0c;使得多…

怒刷LeetCode的第4天(Java版)

#【中秋征文】程序人生&#xff0c;中秋共享# 目录 第一题 题目来源 题目内容 解决方法 方法一&#xff1a;遍历字符串 方法二&#xff1a;有限状态机&#xff08;Finite State Machine&#xff09; 方法三&#xff1a;正则表达式 第二题 题目来源 题目内容 解决方…

机器学习——决策树/随机森林

0、前言&#xff1a; 决策树可以做分类也可以做回归&#xff0c;决策树容易过拟合决策树算法的基本原理是依据信息学熵的概念设计的&#xff08;Logistic回归和贝叶斯是基于概率论&#xff09;&#xff0c;熵最早起源于物理学&#xff0c;在信息学当中表示不确定性的度量&…

带你了解前后端分离的秘密-Vue【vue入门】

&#x1f3c5;我是默&#xff0c;一个在CSDN分享笔记的博主。&#x1f4da;&#x1f4da; &#x1f31f;在这里&#xff0c;我要推荐给大家我的专栏《Vue》。&#x1f3af;&#x1f3af; &#x1f680;无论你是编程小白&#xff0c;还是有一定基础的程序员&#xff0c;这个专栏…

js中事件委托和事件绑定之间的区别

聚沙成塔每天进步一点点 ⭐ 专栏简介⭐ 事件绑定&#xff08;Event Binding&#xff09;⭐事件委托&#xff08;Event Delegation&#xff09;⭐ 选择事件绑定或事件委托⭐ 写在最后 ⭐ 专栏简介 前端入门之旅&#xff1a;探索Web开发的奇妙世界 记得点击上方或者右侧链接订阅本…

NVM安装及如何使用NVM

NVM是什么&#xff1f; nvm 全名 Node Version Manager&#xff0c;Node的版本管理工具 NVM能做什么&#xff1f; 安装 nvm 后&#xff0c;可以使用nvm的相关命令来管理和切换不同的 node 版本&#xff0c;方便开发 如何安装NVM 链接: NVM GitHub地址 如何使用 NVM 命令 …

R语言绘制PCA双标图

代码&#xff1a; setwd("D:/Desktop/0000/R") #更改路径#导入数据 df <- read.table("Input data.csv", header T, sep ",")# ----------------------------------- #所需的包: packages <- c("ggplot2", "tidyr"…

1.简单工厂模式

UML类图 代码 main.cpp #include <iostream> #include "OperationFactory.h" using namespace std;int main(void) {float num1;float num2;char operate;cin >> num1 >> num2 >> operate;Operation* oper OperationFactory::createOpera…

算法综合篇专题四:前缀和

"回忆里的我&#xff0c;比国王富有。奢侈的快乐~" 1、前缀和【模板】 (1) 题目解析 (2) 算法原理 #include <iostream> using namespace std;const int N 100010; // 可能出现溢出 long long arr[N],dp[N]; int n,q;int main() {cin >> n …