【C++】内存管理深入解析

news2024/9/23 11:19:50

目录

  • 1. 内存的五大区域
    • 1.1 栈区(Stack)
    • 1.2 堆区(Heap)
    • 1.3 全局/静态存储区
    • 1.4 常量存储区
    • 1.5 代码区
  • 2. 回顾c语言的动态内存管理
    • 2.1 malloc/calloc/realloc
    • 2.2 free
  • 3. C++中的新旧对话
    • 3.1 new
    • 3.2 delete
  • 4. new/delete的实现原理
    • 4.1 new的工作流程
    • 4.2 delete的工作流程
    • 4.3 new T[N]和delete[ ] 的原理的原理
  • 5.定位new(Placement new)
  • 6. 常见面试题
    • 6.1 malloc/free和new/delete的区别
    • 6.2 内存泄露
  • 7.总结:

在C++的世界里,内存管理是一个至关重要的话题。与C语言相比,C++引入了对象的概念,使得内存管理变得更加复杂但也更加灵活。本文将深入探讨C++中的内存管理机制,包括内存的分布、动态内存的申请与释放,以及new和delete的使用。

1. 内存的五大区域

1.1 栈区(Stack)

栈区主要用于存储函数的局部变量、函数参数等。这部分内存的分配和释放是自动进行的,遵循先进后出的原则。

代码示例

void function() {
    int localVar = 10; // 局部变量存储在栈区
}

1.2 堆区(Heap)

堆区用于程序运行时的动态内存分配,由程序员通过new、delete(C++)或malloc、free(C)手动管理。

代码示例

int* dynamicArray = new int[10]; // 动态分配数组存储在堆区
delete[] dynamicArray; // 释放堆区内存

1.3 全局/静态存储区

全局变量和静态变量(包括静态全局变量和静态局部变量)存储在此区域。这部分内存在程序启动时分配,在程序结束时释放。

代码示例

int globalVar = 20; // 全局变量
static int staticGlobalVar = 30; // 静态全局变量

void function() {
    static int staticLocalVar = 40; // 静态局部变量
}

1.4 常量存储区

常量字符串和其他常量数据存储在此区域。这部分内存通常是只读的。

代码示例

const char* constString = "Hello, World!"; // 常量字符串存储在常量存储区

1.5 代码区

存储程序的二进制代码,即编译后的机器语言指令。这部分内存也是只读的,确保程序代码不会被程序本身或其他程序意外修改。

代码区直接对应于程序的可执行代码

补充说明:

  • 内存映射段:用于映射外设或文件进内存的特殊区域,也可以用于进程间通信。
  • 内核空间:操作系统保留的内存区域,不可直接访问。

2. 回顾c语言的动态内存管理

2.1 malloc/calloc/realloc

malloc函数: 通常用于动态创建数组或者结构体等数据结构。

int *ptr;

ptr = (int*)malloc(5 * sizeof(int));

上述示例代码演示了使用malloc函数动态分配了一个包含5个整数元素的数组,并将其地址赋给指针ptr。

可以将malloc比喻成一个灵活调度仓库管理员,根据不同货物大小和数量进行合理调度并返回货物位置信息.

其中: malloc申请的空间都是未初始化的,即被编译器置为随机值


calloc函数: 在内存中分配指定数量的连续空间,并将每个字节都初始化为零

与malloc函数不同,calloc函数需要两个参数:所需元素的个数每个元素的大小。这使得calloc函数在申请数组等数据结构时更加方便。

int *ptr;

ptr = (int*)calloc(5, sizeof(int));

上述示例代码演示了使用calloc函数动态分配了一个包含5个整数元素的数组,并将其地址赋给指针ptr。与malloc不同,calloc会将分配的内存空间全部初始化为0。

calloc函数用途

  • 用于动态分配数组或结构体等数据结构。

  • 在需要清零初始化内存内容时使用


realloc函数: 用于重新调整动态内存分配大小的函数。它可以改变之前分配区域的大小,同时保留原有内容。

注意:如果新申请的空间比原来的大,则新空间中新增部分未初始化;如果新申请空间比原来小,则原有内容可能会被截断。

int *ptr;

// 假设ptr已经通过malloc或者calloc进行了动态内存分配
ptr = (int*)realloc(ptr, 10 * sizeof(int));

上述示例代码演示了使用realloc函数重新调整了之前已经动态分配过的内存空间,将其扩展为包含10个整数元素的数组。

realloc函数用途

  • 用于调整之前动态分配内存空间的大小。

  • 在需要扩大或缩小已分配内存空间时使用


2.2 free

free函数是用于释放动态分配内存空间的函数。它的作用是将之前通过malloc、calloc或realloc等函数动态分配的内存空间进行释放,以便系统可以重新利用这些空间。

int *ptr;

// 假设ptr已经通过malloc、calloc或者realloc进行了动态内存分配
free(ptr);

注意同一块空间不能释放两次。free 后通常会把指针置空。

3. C++中的新旧对话

C语言中管理函数只能对内置类型使用,而 C++ 中存在很多自定义类型,malloc等函数无能为力。

于是C++引入了new和delete操作符,与C语言的malloc和free相比,它们提供了类型安全和对象构造/析构的自动化

3.1 new

new操作符是C++中用于动态内存分配的关键操作符之一,它与malloc函数类似,但提供了更加便捷和安全的内存管理方式。
使用new操作符可以动态地在堆区分配所需大小的内存空间,并返回该空间的首地址。

与malloc不同

  • new操作符会自动计算所需空间的大小,并且在分配失败时抛出异常,而不是返回NULL指针,如下:
int* p2 = (int*)malloc(sizeof(int));
if (p2 == nullptr)
{
	perror("malloc fail");
}

int *ptr = new int; // 动态创建一个整数对象

malloc后还需要进行判断,而new在分配失败时抛出异常,因为不需要判断。

  • new能在分配后进行初始化
int* p1 = new int(10);	//申请一个int,初始化10
int* p2 = new int[10] {1, 2, 3, 4};.//申请10个int,并初始化
  • new可以用于分配自定义类型
//假设存在A类
A* ptr = new A[5];	//申请五个A类,这些类都在堆上
  • new动态开辟时,会调用自定义类型的构造函数

3.2 delete

使用delete操作符可以释放之前通过new操作符动态分配的内存空间,同时也会调用该内存空间对应对象的析构函数来进行资源清理.

int *ptr = new int; // 动态创建一个整数对象

int *arr = new int[5]; // 动态创建包含5个整数元素的数组

// 在不再需要这些动态分配内存时使用delete进行释放

delete ptr;

delete[] arr;
  • delete调用销毁时,会先调用自定义类型的析构函数

注意:切记,申请与释放要配套使用。malloc和free配套使用,new和delete配套使用


4. new/delete的实现原理

在C++中,new和delete不仅是关键字,它们背后的实现涉及到一系列复杂的操作,包括内存分配、对象构造、内存释放、和对象析构。更深入地,new和delete实际上是通过调用全局函数operator new和operator delete来完成其工作的。

4.1 new的工作流程

当你使用new操作符时,C++会执行以下步骤:

  • 内存分配:首先,new通过调用operator new函数分配足够的内存来存储指定类型的对象。这个函数的默认实现使用malloc来分配内存,但可以被重载。
  • 对象构造:分配内存后,new在分配的内存上调用对象的构造函数来初始化对象。

来看看 operator new 的代码实现:

/*
operator new:该函数实际通过malloc来申请空间,当malloc申请空间成功时直接返回;申请空间失败,尝试执行空间不足应对措施,如果改应对措施用户设置了,则继续申请,否
则抛异常。
*/
void* __CRTDECLoperatornew(size_tsize)_THROW1(_STDbad_alloc)
{
	// try to allocate size bytes
	void* p;
	while ((p = malloc(size)) == 0)
		if (_callnewh(size) == 0)
		{
			// report no memory
			//如果申请内存失败了,这里会抛出bad_alloc类型异常
			staticconststd::bad_allocnomem;
			_RAISE(nomem);
		}
	return(p);
}

其实 operator new 就是通过对 malloc 的封装实现的,不过进行了改进,当对象为自定义类型时,会去调用它的构造函数,并且当开辟失败时,会抛出异常。

4.2 delete的工作流程

与new相对应,当你使用delete操作符时,C++会执行以下步骤:

  • 对象析构:首先,delete调用对象的析构函数来执行任何必要的清理工作。
  • 内存释放:析构函数调用后,delete通过调用operator delete函数释放对象占用的内存。这个函数的默认实现使用free来释放内存,但同样可以被重载。

看看 operator delete 的代码实现

/*
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_FINALLY
		return;
}
/*
free的实现
*/
#define free(p) _free_dbg(p, _NORMAL_BLOCK)

operator delete 也是对 free 进行了封装,改进的就是:当释放对象为自定义类型时,会调用它的析构函数。

4.3 new T[N]和delete[ ] 的原理的原理

new T[N]

  • 1、调用operator new[ ]函数,在operator new[ ]函数中实际调用operator new函数完成N个对象空间的申请。
  • 2、在申请的空间上执行N次构造函数。

delete[ ]

  • 1、在空间上执行N次析构函数,完成N个对象中资源的清理。
  • 2、调用operator delete[ ]函数,在operator delete[ ]函数中实际调用operator delete函数完成N个对象空间的释放。

5.定位new(Placement new)

除了标准的new和delete,C++还提供了定位new(Placement new)的概念,允许在已分配的内存上构造对象。这在需要在特定位置构造对象,或者使用内存池时非常有用。

char buffer[sizeof(MyClass)];
MyClass* myObject = new (buffer) MyClass(); // 在buffer指定的内存上构造对象

// 使用定位new时,不应使用普通的delete来释放内存,因为内存不是通过new分配的
// 相反,应直接调用析构函数
myObject->~MyClass();

在实际的C++编程中,定位new操作符通常用于以下场景:

  • 需要在已分配内存空间的特定位置上构造对象。

  • 实现自定义的内存管理策略,如内存池等

内存池中会预先存放许多从堆中申请过来的空间,这些空间是已经分配好的,需要的时候可以直接用,因此定位new可以在此一展身手。


6. 常见面试题

6.1 malloc/free和new/delete的区别

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

6.2 内存泄露

  • 内存泄漏:

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


  • 内存泄漏的危害:

长期运行的程序出现内存泄漏,影响很大,如操作系统、后台服务等等,出现内存泄漏会导致响应越来越慢,最终卡死。

void MemoryLeaks()
{
	// 1.内存申请了忘记释放
	int* p1 = (int*)malloc(sizeof(int));
	int* p2 = new int;

	// 2.异常安全问题
	int* p3 = new int[10];
	Func(); // 这里Func函数抛异常导致 delete[] p3未执行,p3没被释放.
	delete[] p3;
}


  • 内存泄漏分类
    在C/C++中我们一般关心两种方面的内存泄漏:

1、堆内存泄漏(Heap Leak)

堆内存指的是程序执行中通过malloc、calloc、realloc、new等从堆中分配的一块内存,用完后必须通过调用相应的free或者delete释放。假设程序的设计错误导致这部分内容没有被释放,那么以后这部分空间将无法再被使用,就会产生Heap
Leak。

2、系统资源泄漏

指程序使用系统分配的资源,比方套接字、文件描述符、管道等没有使用对应的函数释放掉,导致系统资源的浪费,严重可导致系统效能减少,系统执行不稳定。


  • 如何避免内存泄漏?
     1、工程前期良好的设计规范,养成良好的编码规范,申请的内存空间记住匹配的去释放。
     2、采用RAII思想或者智能指针来管理资源。
     3、有些公司内部规范使用内部实现的私有内存管理库,该库自带内存泄漏检测的功能选项。
     4、出问题了使用内存泄漏工具检测。

  • 内存泄漏非常常见,解决方案分为两种:
     1、事前预防型。如智能指针等。
     2、事后查错型。如泄漏检测工具。

7.总结:

C++的动态内存管理是一个强大的特性,允许开发者在运行时分配和释放内存。然而,这也带来了额外的责任,如内存泄漏、野指针和重复释放内存等问题。通过遵循最佳实践和利用C++提供的工具,如智能指针和标准库容器,开发者可以有效地管理内存,写出更稳定、更高效的代码。

在这里插入图片描述

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

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

相关文章

IDEA生成可执行jar包

1. 进入需要打包的项目,选择 最上方菜单栏的 File → Project Structure 2. 选择 左侧菜单栏 Artifacts → 加号 → JAR → from modules with dependencies 3. 选择入口类 Main Class(点击文件夹图标可以快速选择),点击 OK&#…

(四)elasticsearch 源码之索引流程分析

https://www.cnblogs.com/darcy-yuan/p/17024341.html 1.概览 前面我们讨论了es是如何启动,本文研究下es是如何索引文档的。 下面是启动流程图,我们按照流程图的顺序依次描述。 其中主要类的关系如下: 2. 索引流程 (primary) 我们用postman发送请求&…

【漏洞复现】EPON上行A8-C政企网关未授权下载漏洞

Nx01 产品简介 EPON上行A8-C政企网关是一款终端产品,提供企业网络解决方案。 Nx02 漏洞描述 EPON上行A8-C政企网关配置文件未授权下载漏洞,攻击者在未授权状态下下载配置文件,获取配置文件内敏感信息。 Nx03 产品主页 fofa-query: "Z…

龙宝宝头像怎么做?分享3个AI生成工具!

龙宝宝头像怎么做?分享3个AI生成工具! 在这个数字化时代,社交媒体已成为我们展示个性和表达自我的重要舞台。龙宝宝头像不仅有传统的中国龙形象,还有各种创意十足的卡通头像。然而,设计一个理想的头像并非易事&#x…

【面试】零基础自学Java路线及课程推荐,适用各阶段人群

目前各个平台上都有着各种各样的教程,给我们带来了更多选择的同时,也为新手小白带来了一丢丢的麻烦:如何系统学习Java?选择什么样的方式学习?本篇就来解答这些问题。 一 学习方式的选择 学习计算机的主流方式主要是视频课程、书…

22.HarmonyOS App(JAVA)位置布局PositionLayout使用方法

不常用 在PositionLayout中,子组件通过指定准确的x/y坐标值在屏幕上显示。(0, 0)为左上角;当向下或向右移动时,坐标值变大;允许组件之间互相重叠 布局方式 PositionLayout以坐标的形式控制组件的显示位置,允许组件相…

小白买新电脑保姆验机教程

目录 前言: 正文: 7,查看硬件信息 8,检查硬盘 9,检查屏幕 10,烤机检测 总结: 前言: 最近因为学习的需求,购入了一台新的笔记本,正巧趁这次给身边的朋友…

《动手学深度学习(PyTorch版)》笔记6.2

注:书中对代码的讲解并不详细,本文对很多细节做了详细注释。另外,书上的源代码是在Jupyter Notebook上运行的,较为分散,本文将代码集中起来,并加以完善,全部用vscode在python 3.9.18下测试通过&…

显示器校准软件:BetterDisplay Pro for Mac v2.0.11激活版下载

BetterDisplay Pro是一款由waydabber开发的Mac平台上的显示器校准软件,可以帮助用户调整显示器的颜色和亮度,以获得更加真实、清晰和舒适的视觉体验。 软件下载: BetterDisplay Pro for Mac v2.0.11激活版下载 以下是BetterDisplay Pro的主要…

没有联合和枚举 , C语言怎么能在江湖混 ?

本篇会加入个人的所谓‘鱼式疯言’ ❤️❤️❤️鱼式疯言:❤️❤️❤️此疯言非彼疯言 而是理解过并总结出来通俗易懂的大白话, 我会尽可能的在每个概念后插入鱼式疯言,帮助大家理解的. 🤭🤭🤭可能说的不是那么严谨.但小编初心是能让更多人能…

代码随想录算法训练营DAY14 | 二叉树 (1)

一、二叉树理论基础 1.存储方式 链式存储: 顺序存储: 2.二叉树标准定义(Java) public class TreeNode {int val;TreeNode left;TreeNode right;TreeNode() {}TreeNode(int val) { this.val val; }TreeNode(int val, TreeNode left, TreeNode right) {…

带你实现用自己域名打开Tomcat

文章目录 Tomcat1.1、Tomcat 下载1.2、Tomcat 文件图解1.3、 启动或关闭 Tomcat1.3.1、 启动1.3.2、 关闭程序2.1、 修改端口号2.2、修改主机名称Tomcat 1.1、Tomcat 下载 首先去Tomcat 官网下载找到我们需要下载的版本 1.2、To

SpringBoot:多环境配置

多环境配置demo代码:点击查看LearnSpringBoot02 点击查看更多的SpringBoot教程 方式一、多个properties文件配置 注意:创建properties文件,命名规则:application-(环境名称) 示例:application-dev.proper…

c#cad 创建-点(六)

运行环境 vs2022 c# cad2016 调试成功 一、代码说明 创建一个点的命令方法。代码的主要功能是在当前活动文档中创建一个点,并将其添加到模型空间块表记录中。 代码的主要步骤如下: 获取当前活动文档、数据库和编辑器对象。使用事务开始创建点的过程…

Java设计模式大全:23种常见的设计模式详解(二)

本系列文章简介: 设计模式是在软件开发过程中,经过实践和总结得到的一套解决特定问题的可复用的模板。它是一种在特定情境中经过验证的经验和技巧的集合,可以帮助开发人员设计出高效、可维护、可扩展和可复用的软件系统。设计模式提供了一种在…

Leetcode第123场双周赛

Leetcode第123场双周赛 本人水平有限,只做前三道 一、三角形类型 给你一个下标从 0 开始长度为 3 的整数数组 nums ,需要用它们来构造三角形。 如果一个三角形的所有边长度相等,那么这个三角形称为 equilateral 。 如果一个三角形恰好有两…

【lesson11】高并发内存池性能优化

文章目录 高并发内存池性能问题基数树优化性能代码一层基数树两层基数树三层基数树 一层基数树替代mapPageCache.hPageCache.cpp基数树线程安全的原因 高并发内存池性能问题 我们知道,我们实现的高并发内存池存在大量的申请锁和,释放锁,而这…

【RL】Basic Concepts in Reinforcement Learning

Lecture1: Basic Concepts in Reinforcement Learning MDP(Markov Decision Process) Key Elements of MDP Set State: The set of states S \mathcal{S} S(状态 S \mathcal{S} S的集合) Action: the set of actions A ( s ) \mathcal{A}(s) A(s)…

奚梦瑶何猷君香港共度佳节,幸福全家福彰显深厚亲情。

♥ 为方便您进行讨论和分享,同时也为能带给您不一样的参与感。请您在阅读本文之前,点击一下“关注”,非常感谢您的支持! 文 |猴哥聊娱乐 编 辑|徐 婷 校 对|侯欢庭 从奚梦瑶父母与赌王家族的全家福中,我们可感受到两…

图灵之旅--二叉树堆排序

目录 树型结构概念树的表示形式 二叉树概念特殊的二叉树二叉树性质二叉树的存储二叉树的遍历前中后序遍历 优先级队列(堆)概念 优先级队列的模拟实现堆的性质概念堆的存储方式堆的创建 堆常用接口介绍PriorityQueue的特性PriorityQueue常用接口介绍优先级队列的构造插入/删除/获…