C++动态内存管理:与C语言动态内存管理的差异之争

当你改错一行代码的时候:

当你想要重构别人的代码时:

目录

前言

一、C/C++的内存分布

二、C/C++语言中的动态内存管理

 三、new与delete的实现原理

总结:

 


前言

在C++中,内存管理是一个至关重要的主题。正确地管理内存可以避免内存泄漏和内存访问错误等问题,提高程序的性能和可靠性。本文将深入探讨C++中的内存管理机制,包括从C语言内存管理到C++内存管理转变,二者的关联等内容。

一、C/C++的内存分布

C/C++程序内存主要使用了以下几个区域,让我们先来认识一下:

1、内核区域:

“内核区域”通常指的是操作系统的核心部分,它是操作系统的基本组成部分之一。内核负责管理计算机的硬件资源,并提供给应用程序访问这些资源的接口。但是我们用户的代码读写不了它,所以这里也不主要讲解。

2、栈区:

栈区是计算机内存中的一部分,用于存储函数调用时的局部变量、函数参数、函数返回地址等临时数据。栈是一种后进先出(LIFO)的数据结构,因此栈区也被称为后进先出(LIFO)内存区域。

在典型的程序执行过程中,每当一个函数被调用时,该函数的局部变量和参数被分配到栈区,并且该函数的返回地址被推入栈中。当函数执行完成后,栈中的这些数据被销毁,栈的指针回退到上一个函数的栈帧。这种方式使得程序能够有效地管理函数调用和返回。

栈区的大小通常是固定的,由操作系统在程序启动时分配。如果栈区的空间被耗尽,就会发生栈溢出(stack overflow)错误,这通常是由于递归函数调用层数过多或者局部变量占用过多栈空间所导致的。

3、内存映射段:

存映射段(Memory-mapped segment)是指操作系统中的一种机制,它允许将文件或其他设备映射到进程的地址空间,使得这些文件或设备能够像内存一样被访问和操作。(我们这里也不讲解,只是单纯提一下)

4、堆区:

堆区是计算机内存中的一部分,用于动态分配内存空间。在堆区中,程序员可以通过调用诸如 malloc()、calloc() 或 new 等函数来请求内存空间,用于存储程序中动态创建的数据结构,比如链表、树、对象等。(也就是我们今天所要讲解的动态内存管理所涉及的区域)

5、数据段:

数据段(Data Segment)是指存储程序中已声明的全局变量和静态变量的内存区域。数据段通常位于进程的虚拟地址空间中的一个固定位置,并且在程序加载时被分配。(全局变量与static修饰的变量)

(静态变量主要有三种方式:

1、在函数内部使用 static 关键字声明

2、在全局作用域内使用 static 关键字声明

3、在函数内部不使用 static 关键字,但函数外部使用 static 修饰该函数

6、代码段:

代码段中通常存储着可执行的程序与只读常量如我们规定的“a,n,f,j”等字符

二、C/C++语言中的动态内存管理

1、C语言:

在C语言中,我们通常使用malloc/calloc/realloc/free四个关键字来管理我们的动态内存分配。

malloc()函数:

  • void *malloc(size_t size);
  • 用于动态分配指定大小的内存块。
  • 返回一个指向分配的内存块的指针,或者在分配失败时返回 NULL。

calloc()函数 :

  • void *calloc(size_t num, size_t size);
  • 用于动态分配指定数量和大小的内存块,并将其初始化为零。
  • 返回一个指向分配的内存块的指针,或者在分配失败时返回 NULL。

realloc()函数;

  • void *realloc(void *ptr, size_t size);
  • 用于更改之前分配的内存块的大小。
  • 返回一个指向重新分配的内存块的指针,或者在分配失败时返回 NULL。

 free()函数:

  • void free(void *ptr);
  • 用于释放之前分配的内存块。

 在C语言中,malloc()等函数会返回 void* 类型的指针,需要强制类型转换为目标类型,这使得使用十分不方便。而C++内存管理只需要用new与delete两个运算符就可以做得更加高级,更加安全便捷。

2、C++:
C++中用 newdelete 运算符来动态地分配和释放内存。

new运算符:

  • new 运算符用于动态分配内存空间,并返回所分配内存的地址。
  • new 运算符的基本语法为:new 数据类型new 数据类型[数组大小]
  • 动态分配的内存必须由程序员显式地释放,否则会造成内存泄漏。

delete运算符:

  • delete 运算符用于释放动态分配的内存空间。
  • delete 运算符的基本语法为:delete 指针变量delete[] 指针变量(释放数组)。
  • 动态分配的内存在不再使用时必须由程序员显式地释放,否则会造成内存泄漏。

 (其实在C++11中还新增了智能指针的概念,但现在我们先不提及)

int main()
{
	int* arr = new int[10];// 分配一个包含10个整数的内存块
	int* ptr = new int; // 动态分配一个整数的内存空间
	int* ptr2 = new int(10); // 动态分配一个整数的内存空间并且初始化为10

	delete ptr;
	delete ptr2;//释放内存
	delete[] arr;//加上[]表示多个数据

	return 0;
}

//分配自定义类型

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

int main()
{
	A* ptr1 = new A;
	A* ptr2 = (A*)malloc(sizeof(A));

	delete ptr1;
	free(ptr2);
	return 0;
}
结果可以发现,我们在new与delete的时候会自动调用自定义类型的构造与析构函数,而C的malloc与free则不会。
new delete 是用户进行 动态内存申请和释放的操作符 operator new operator delete
系统提供的 全局函数 new 在底层调用 operator new 全局函数来申请空间, delete 在底层通过
operator delete 全局函数来释放空间。

实际上,operator new 实际也是通过malloc来申请空间,如果malloc申请空间成功就直接返回,否则执行用户提供的空间不足应对措施,如果用户提供该措施就继续申请,否则就抛异常。(以后讲解异常概念)operator delete 最终是通过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[] 的原理
1. 在释放的对象空间上执行 N 次析构函数,完成 N 个对象中资源的清理
2. 调用 operator delete[] 释放空间,实际在 operator delete[] 中调用 operator delete 来释
放空间

总结:

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

 

希望通过本篇文章,能够对你区分C++与C语言内存管理有所帮助。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

上海AI Lab开源首个可替代GPT-4V的多模态大模型

与开源和闭源模型相比&#xff0c;InternVL 1.5 在 OCR、多模态、数学和多轮对话等 18 个基准测试中的 8 个中取得了最先进的结果。 上海AI Lab 推出的 InternVL 1.5 是一款开源的多模态大语言模型 (MLLM)&#xff0c;旨在弥合开源模型和专有商业模型在多模态理解方面的能力差距…

二、SPI协议

文章目录 总述1.SPI接口2. SPI工作模式3. SPI通信时序4. SPI协议 对比 UART协议&#xff08;上一篇文章刚介绍过uart协议&#xff0c;这里来对比一下&#xff09; 总述 SPI&#xff08;Serial Peripheral Interface&#xff09;是一种高速的、全双工、同步的串行通信总线&…

【影片欣赏】【指环王】【魔戒:国王归来 The Lord of the Rings: The Return of the King】

往期魔戒博客见&#xff1a; 【影片欣赏】【指环王】【魔戒&#xff1a;护戒使者 The Lord of the Rings: The Fellowship of the Ring】 【影片欣赏】【指环王】【魔戒&#xff1a;双塔奇谋 The Lord of the Rings: The Two Towers】 2004年发行&#xff0c;Special Extend…

副业兼职没那么难,视频号带货,1天稳定500,适合新手操作

向大家推荐一个项目&#xff1a;视频号书单号带货玩法。我已经实践了一段时间&#xff0c;并成功售出了1200多单&#xff0c;赚取了2万多元。这个项目表现相当出色&#xff0c;强烈推荐给大家&#xff01; 周周近财&#xff1a;让网络小白少花冤枉钱&#xff0c;赚取第一桶金 …

Linux vscode push报错fatal: Authentication failed

注意啊&#xff0c;Git基于密码的身份验证已经被删除了&#xff0c;所以这个报错发生时无论密码正确与否&#xff0c;以及参考比较旧的改bug教程&#xff0c;都没法提交。进入提示的网址&#xff0c;生成个人访问令牌就好了

200-500人规模工厂网络方案(中小企业网络)

一、方案概述 工厂一般有单独的弱电房&#xff0c;类似这种 里面采用的方案如下&#xff1a; 主要考虑有线、无线、财务、办公、访客等业务&#xff0c;便于维护管理和后续扩容 还需要 Wi-Fi覆盖零死角高速率&#xff0c;工作不卡顿 同时考虑AV反病毒、IPS入侵防御、用户准…

【MySQL数据库开发设计规范】之命名规范

欢迎点开这篇文章&#xff0c;自我介绍一下哈&#xff0c;本人姑苏老陈 &#xff0c;是一名JAVA开发老兵。 本文收录于 《MySQL数据库开发设计规范》专栏中&#xff0c;该专栏主要分享一些关于MySQL数据库开发设计相关的技术规范文章&#xff0c;定期更新&#xff0c;欢迎关注&…

python自动化生成ppt

使用Python和python-pptx创建PPT 在这篇博客中&#xff0c;我们将探讨如何使用Python库python-pptx来创建一个简单的PowerPoint演示文稿&#xff08;PPT&#xff09;。这个库允许我们以编程方式创建幻灯片、添加文本、图片、表格和自定义形状。 安装python-pptx 首先&#x…

智能助手上线,大模型提供云服务专属顾问

业务背景 在使用云服务的时候&#xff0c;当您遇到复杂问题&#xff0c;如配置、关联或计费方式不明确时&#xff0c;可能需要向客服提交工单进行技术沟通。在漫长的工作过程中&#xff0c;耗费了宝贵的时间和精力。 2024 年 4 月&#xff0c;百度智能云正式推出了融合文心大…

单调栈:(C++)

在题目的要求中&#xff0c;存在先进后出&#xff08;即在前面的数据需要遍历到后面的某一数据时才能确定计算值&#xff09;单调栈在一部分解题场景中避免了暴力解法的高时间复杂度问题&#xff0c;但是在做题过程中视情况而定&#xff0c;有些题目的最优解不一定使用单调栈&a…

【原创】springboot+mysql物资库存管理系统设计与实现

个人主页&#xff1a;程序猿小小杨 个人简介&#xff1a;从事开发多年&#xff0c;Java、Php、Python、前端开发均有涉猎 博客内容&#xff1a;Java项目实战、项目演示、技术分享 文末有作者名片&#xff0c;希望和大家一起共同进步&#xff0c;你只管努力&#xff0c;剩下的交…

利用AI提高内容生产效率的五个方案

目录 如何利用AI提高内容生产效率? ​编辑方向一&#xff1a;自动化内容生成 方向二&#xff1a;内容分发与推广 方向三&#xff1a;内容分析与优化 方向四&#xff1a;图像和音频处理 方向五&#xff1a;自动编辑和校对 如何利用AI提高内容生产效率? 简介&#xff1a…

车载电子电器架构 —— 应用软件开发(上)

车载电子电器架构 —— 应用软件开发(上) 我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 屏蔽力是信息过载时代一个人的特殊竞争力,任何消耗你的人和事,多看一眼都是你的不对。非必要不费力证明…

CellChat包文献介绍

Inference and analysis of cell-cell communication using CellChat - PubMed (nih.gov) 目录 在线数据 摘要 基础介绍 分析结果 1&#xff0c;概述 2&#xff0c;识别预测通路 3&#xff0c;连续的信号转导 4&#xff0c;预测空间共定位细胞群之间的关键信号转导事件…

企业活动想联系媒体报道宣传如何联系媒体?

在企业的宣传推广工作中,我曾经历过一段费事费力、效率极低的时期。那时,每当公司有重要活动或新项目需要媒体报道时,我便要一家家地联系媒体,发送邮件、打电话,甚至亲自登门拜访,只为求得一篇报道。然而,这样的过程充满了不确定性和挑战,时常让我感到焦虑和压力山大。 记得有一…

vue3专栏项目 -- 项目介绍以及准备工作

这是vue3TS的项目&#xff0c;是一个类似知乎的网站&#xff0c;可以展示专栏和文章的详情&#xff0c;可以登录、注册用户&#xff0c;可以创建、删除、修改文章&#xff0c;可以上传图片等等。 这个项目全部采用Composition API 编写&#xff0c;并且使用了TypeScript&#…

亚马逊产品排名提升全攻略:自养号测评干货

之前我们一同探讨了亚马逊产品排名的多种类型&#xff0c;现在让我们回到正题&#xff0c;探讨一下如何才能有效地提升产品排名&#xff0c;从而吸引并抓住平台的流量&#xff0c;最终将其转化为可观的销量。 首先&#xff0c;卖家必须明晰亚马逊的排名机制&#xff0c;它主要基…

网页版Figma汉化

最近学习Figma&#xff0c;简单介绍一下网页版Figma的汉化方法 1.打开网址&#xff1a;Figma软件汉化-Figma中文版下载-Figma中文社区 2.下载汉化插件离线包 解压汉化包 3.点开谷歌的管理扩展程序 4.点击加载已解压的扩展程序&#xff0c;选择刚刚解压的包 这样就安装好了汉化…

从0到1开发一个vue3+ts项目(一)

1. 环境配置 1.1 安装node 使用官方安装程序 前往 Node.js 官网&#xff1a;访问 Node.js 官网&#xff0c;下载适合你操作系统的安装程序。运行安装程序&#xff1a;下载完成后&#xff0c;双击安装程序并按照提示进行安装。验证安装&#xff1a;安装完成后&#xff0c;在终…

顺序表经典算法OJ题-- 力扣27,88

题1&#xff1a; 移除元素 题2&#xff1a; 合并两个有序数组 一&#xff1a;题目链接&#xff1a;. - 力扣&#xff08;LetCode&#xff09; 思路&#xff1a;&#xff08;双指针法&#xff09; 创建两个变量src&#xff0c;dst 1&#xff09;若src指向的值为val&#xf…