【C++语言】C/C++内存管理

一、C/C++内存分布

       我们先来看一看C/C++中有哪些区域,为什么C/C++中区分这些区域呢??不同的数据有不同的存储需求,各个区域满足不同的需求。我们有临时用的数据,该数据是存储在栈帧区域的;在一些数据结构中,算法中需要动态开辟一些空间,比如:归并排序,这些数据是存储在堆区的;在整个程序期间都要使用的数据,我们称之为只读数据,像全局数据、静态数据存储在静态区中,像可执行代码和只读常量存储在常量区中。

我们下面来总结一下:

  1. 栈又叫做堆栈——非静态局部变量/函数参数/返回值等等,栈是向下增长的
  2. 内存映射段是高效的I/O映射方式,用于装载到一个共享的动态内存库,用户可以使用系统接口创建共享内存,进行进程间通信
  3. 堆用于程序运行时动态内存分配,堆是可以向上增长的
  4. 数据段(静态区)——存储全局数据和静态数据
  5. 代码段(常量区)——可执行的代码和只读常量

1.2 练习习题

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

选择题:

选项:A.栈     B.堆    C.数据段(静态区)  D.代码段(常量区)

  • globalVar在哪里?  全局变量,在静态区  C
  • staticGlobalVar在哪里? 全局变量,用static修改,静态变量,在静态区 C
  • staticVar在哪里?  用static修饰,静态变量,但是作用范围被限制在函数中  在静态区 C
  • localVar在哪里?  局部变量,在栈区  A
  • num1 在哪里?  局部变量,在栈区  A
  • char2在哪里?  局部变量,在栈区开辟数组,在栈区  A
  • *char2在哪里?  局部变量,指针指向的地方在栈区 A
  • pChar3在哪里?  数组指针,局部变量,在栈区 A
  • *pChar3在哪里?  字面值常量,只读常量,在常量区  D
  • ptr1在哪里?  指针,局部变量,在栈区  A
  • *ptr1在哪里?  利用malloc函数开辟空间,在堆区  B

填空题:

  • sizeof(num1) =  40
  • sizeof(char2) =  5
  • strlen(char2) =  4
  • sizeof(pChar3) =  4/8
  • strlen(pChar3) =  4
  • sizeof(ptr1) =  4/8

二、C++内存管理方式

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

2.1 new/delete操作内置类型

void Test()
{// 动态申请一个int类型的空间int *ptr4 = new int;// 动态申请一个int类型的空间并初始化为10int *ptr5 = new int(10);// 动态申请10个int类型的空间int *ptr6 = new int[10];delete ptr4;delete ptr5;delete[] ptr6;
}

 

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

2.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不会。

三、operator new 和 operator delete 函数

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

       我们来看一下这两个函数在底层是如何封装的?我们在C语言中使用malloc函数和free函数来封装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 实际上就是使用malloc函数来申请空间的,如果使用malloc函数申请空间成功就直接返回,否则执行用户提供空间不足应对措施,如果用户提供该措施就继续申请,否则就抛异常。operator delete 最终就是通过free来释放空间的。

       为什么我们要重新封装一下malloc函数呢??因为C++语言是面向对象的语言,但是之前的malloc是C语言的,不具有面向对象的思想。因为malloc在创建空间失败后,我们需要抛出一个异常,但是malloc函数本身是返回0,所以我们需要将malloc函数重新封装满足我们面向对象的思想。封装free函数其实没有什么,为了对称,我们需要进行重新封装free函数。

我们在使用这些函数,我们需要进行配对使用,如果我们将类中的析构函数进行注释,(这里有一些重点)

四、new和delete的实现原理

4.1 内置类型

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

4.2 自定义类型

new的原理:

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

delete的原理:

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

new T[N] 的原理:

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

delete[] 的原理:

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

五、定位new表达式

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

使用的格式是:

new(place_address)type 或者 new(place_address)type(initializer-list)
place_address必须是一个指针,initializer-list是类型的初始化列表

使用场景:
       定位new表达式在实际上一般是配合内存池使用的,因为在内存池分配的内存没有初始化,所以如果是自定义类型的对象,需要使用new的定义表达式进行显示地调构造函数进行初始化。

六、常见的面试题

6.1 malloc/free 和 new/delete 的区别(从用法和原理上分析)

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

  • malloc和free是函数,new和delete是操作符
  • malloc申请的空间不会初始化,new可以进行初始化
  • malloc申请空间时,需要手动计算空间大小并传递,new只需在其后面跟上空间的类型即可,如果是多个对象,[]中指定对象个数即可
  • malloc申请空间失败时,返回的是NULL,因此使用时必须判空,new不需要,但是new需要捕获异常

       申请自定义类型对象时,malloc/free只会开辟空间,不会调用构造函数和析构函数,而在new申请空间后会调用构造函数完成对象的初始化,delete在释放空间前会调用析构函数完成空间中资源的清理。

6.2 内存泄露

6.2.1 什么是内存泄露,内存泄露的危害

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

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

6.2.2 内存泄露分类(了解)

C/C++程序中一般我们关心两种方面的内存泄露:

堆内存泄露

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

系统资源泄露

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

6.2.3 如何检测内存泄露(了解)

       在VS下,可以使用windows操作系统提供的_CrtDumpMemoryLeaks()函数进行简单检测,该函数只报出了大概率泄露了多少个字节,没有其他更准确的位置信息。

int main()
{int* p = new int[10];// 将该函数放在main函数之后,每次程序退出的时候就会检测是否存在内存泄漏_CrtDumpMemoryLeaks();return 0;
}// 程序退出后,在输出窗口中可以检测到泄漏了多少字节,但是没有具体的位置
Detected memory leaks!
Dumping objects ->
{79} normal block at 0x00EC5FB8, 40 bytes long.Data: < > CD CD CD CD CD CD CD CD CD CD CD CD CD CD CD CD
Object dump complete.

6.2.4 如何避免内存泄露

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

总结一下:

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

  1. 事前预防型,比如智能指针等
  2. 事后查错型,比如泄露检测工具

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

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

相关文章

『功能项目』回调函数处理死亡【54】

我们打开上一篇53伤害数字UI显示的项目&#xff0c; 本章要做的事情是使用回调函数处理怪物Boss01死亡后增加主角经验值的功能&#xff0c;以及生成一个七秒的升级特效 首先增加一个技能特效重命名为PlayerUpGradeEffect 修改脚本&#xff1a;BossCtrl.cs 修改脚本&#xff1a…

【Linux系统编程】信号的保存与处理

目录 一&#xff0c;信号的保存 1-1&#xff0c;core与Term终止信号 1-2&#xff0c;进程退出与信号的关系 1-3&#xff0c;信号在内核中的表示 1-4&#xff0c;信号操作函数 二&#xff0c;信号的处理 2-1&#xff0c;信号被处理的时期 2-2&#xff0c;内核实现信号的…

zip-password-finder

1.zip-password-finder 对于传统ZIP文件密码的破解&#xff0c;采用密码匹配的方式进行实现&#xff0c;该github库的地址是&#xff1a; GitHub - agourlay/zip-password-finder: Find the password of protected ZIP files.Find the password of protected ZIP files. Cont…

整数二分算法和浮点数二分算法

整数二分算法和浮点数二分算法 二分 现实中运用到二分的就是猜数字的游戏 假如有A同学说B同学所说数的大小&#xff0c;B同学要在1~100中间猜中数字65&#xff0c;当B同学每次说的数都是范围的一半时这就算是一个二分查找的过程 二分查找的前提是这个数字序列要有单调性 基…

java--JDBC-连接池----JDBC小总结

一.连接池 1.连接池概述 目的&#xff1a;为了解决建立数据库连接耗费资源和时间很多的问题&#xff0c;提高性能。 Connection对象在JDBC使用的时候就会去创建一个对象,使用结束以后就会将这个对象给销毁了(close).每次创建和销毁对象都是耗时操作.需要使用连接池对其进行优…

类加载器详细介绍

类加载器我们要聊一个神秘而又重要的角色——Java类加载器。这家伙&#xff0c;就像是个超级英雄&#xff0c;总是在关键时刻挺身而出&#xff0c;为我们的Java程序提供强大的支持。我会尽量用简单易懂的方式来介绍它。 一 、类加载器介绍 1、类加载器是什么&#xff1f; 想象…

【python设计模式2】创建型模式1

目录 简单工厂模式 工厂方法模式 简单工厂模式 简单工厂模式不是23中设计模式中的&#xff0c;但是必须要知道。简单工厂模式不直接向客户端暴露对象创建的细节&#xff0c;而是通过一个工厂类来负责创建产品类的实例。简单工程模式的角色有&#xff1a;工厂角色、抽象产品角…

Redis(redis基础,SpringCache,SpringDataRedis)

文章目录 前言一、Redis基础1. Redis简介2. Redis下载与安装3. Redis服务启动与停止3 Redis数据类型4. Redis常用命令5. 扩展数据类型 二、在Java中操作Redis1. Spring Data Redis的使用1.1. 介绍1.2. 环境搭建1.3. 编写配置类&#xff0c;创建RedisTemplate对象1.4. 通过Redis…

Linux入门学习:git

文章目录 1. 创建仓库2. 仓库克隆3. 上传文件4. 相关问题4.1 git进程阻塞4.2 git log4.3 上传的三个步骤在做什么 本文介绍如何在Linux操作系统下简单使用git&#xff0c;对自己的代码进行云端保存。 1. 创建仓库 &#x1f539;这里演示gitee的仓库创建。 2. 仓库克隆 &…

Zookeeper 3.8.4 安装和参数解析

安装 zookeeper 之前必须先安装 JDK&#xff0c;有关Linux环境JDK可以参考我以前写的博文 1、关于Linux服务器配置java环境遇到的问题 2、Linux环境安装openJDK 3、Centos7.3云服务器上安装Nginx、MySQL、JDK、Tomcat环境 文章目录 1. zookeeper 安装2. 参数解析 1. zookeeper …

VScode快速配置c++(菜鸟版)

1.vscode是什么 Visual Stdio Code简称VS Code&#xff0c;是一款跨平台的、免费且开源的现代轻量级代码编辑器&#xff0c;支持几乎 主流开发语言的语法高亮、智能代码补全、自定义快捷键、括号匹配和颜色区分、代码片段提示、代码对比等特性&#xff0c;也拥有对git的开箱即…

[乱码]确保命令行窗口与主流集成开发环境(IDE)统一采用UTF-8编码,以规避乱码问题

文章目录 一、前言二、命令行窗口修改编码为UTF-8三、Visual Studio 2022修改编码为UTF-8四、Eclipse修改编码为UTF-8五、DevCPP修改编码为UTF-8六、Sublime Text修改编码为UTF-8七、PyCharm、IDEA、VS Code及Python自带解释器修改编码为UTF-8 一、前言 在学习的征途中&#x…

如何通过 4 种方法恢复 Mac 上删除/未保存的 Excel 文件

您花了数小时在 MacBook 上处理 Excel 工作簿&#xff0c;但现在它不见了。或者&#xff0c;当您退出 Excel 文件时&#xff0c;您无意中选择了“不保存”。这是否意味着您的所有努力都白费了&#xff1f;本文系统地解释了如何在 Mac 上恢复丢失的 Excel 文件。使用我们的 4 种…

如何在Android上实现RTSP服务器

技术背景 在Android上实现RTSP服务器确实是一个不太常见的需求&#xff0c;因为Android平台主要是为客户端应用设计的。在一些内网场景下&#xff0c;我们更希望把安卓终端或开发板&#xff0c;作为一个IPC&#xff08;网络摄像机&#xff09;一样&#xff0c;对外提供个拉流的…

Linux学习记录十四----------线程的创建和回收

文章目录 五、Linux线程1.守护进程1.1.守护进程的特点1.2.进程组1.3会话1.4创建守护进程模型 2.线程的概念3.线程的创建及相关函数3.1.创建线程‐‐pthread_create3.2.单个线程退出 --pthread_exit3.3.阻塞等待线程退出&#xff0c;获取线程退出状态--pthread_join3.4.线程分离…

无限制使用OpenAI最新o1-mini、o1-preview模型:经济高效的AI推理模型

OpenAI 最新推出的 o1 模型是该公司推理模型家族的首位成员&#xff0c;它通过创新的“思维链”训练模式&#xff0c;显著提升了逻辑推理和问题解决的能力。o1 模型在编程竞赛问题、数学奥林匹克资格赛以及物理、生物和化学问题的基准测试中表现出色&#xff0c;甚至在某些领域…

学成在线练习(HTML+CSS)

准备工作 项目目录 内部包含当前网站的所有素材&#xff0c;包含 HTML、CSS、图片、JavaScript等等 1.由于元素具有一些默认样式&#xff0c;可能是我们写网页过程中根本不需要的&#xff0c;所有我们可以在写代码之前就将其清除 base.css /* 基础公共样式&#xff1a;清除…

Tongweb7启动的时候显示要输入java参数(by lqw)

问题描述&#xff1a; 启动tongweb7的时候&#xff0c;提示要输入java参数&#xff0c;如下图所示&#xff1a; 原因&#xff1a; tongweb安装目录bin目录下的external.vmoptions文件改动&#xff0c;在# 符号后加多了一个空格。 external.vmoptions记录的是启动参数&#xf…

【读书】原则

后面的 太长了&#xff0c;而且太多了 我看作者 49年的 0多岁的老人的谆谆教诲 太多了 一下子吃不消 分为 生活原则 和 工作原则 倡导 人要以 原则而活 要做到极度透明 极度求真和极度透明&#xff1a;在软件开发中&#xff0c;对事实的执着追求和对信息的透明度是至关重要的。…

论文阅读 - SELF-REFINE: Iterative Refinement with Self-Feedback

https://arxiv.org/pdf/2303.17651 目录 Abstract Introduction 2 Iterative Refinement with SELF-REFINE Evaluation 3.1 Instantiating SELF-REFINE 3.2 Metrics 3.3 Results Abstract 与人类一样&#xff0c;大型语言模型&#xff08;LLMs&#xff09;并非总能在首次…