当前位置: 首页 > news >正文

C++ 第三讲:内存管理

C++ 第三讲:内存管理

  • 1.C++内存分布
  • 2.内存管理方式
    • 2.1C语言内存管理方式
    • 2.2C++内存管理方式
      • 2.2.1new\delete操作内置类型
      • 2.2.2new\delete操作自定义类型
  • 3.operator new与operator delete函数
  • 4.new和delete实现原理
    • 4.1内置类型
    • 4.2自定义类型
  • 5.定位new
    • 5.1内存池的基本了解
    • 5.2定位new的使用方法
    • 5.3new和delete使用注意事项
  • 6.malloc\free和new\delete的区别

1.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在哪里?____

解答:

int globalVar = 1;//全局变量,应该存放在静态区
static int staticGlobalVar = 1;//静态变量,放在静态区
void Test()
{static int staticVar = 1;//静态数据,静态区int localVar = 1;//变量,创建在栈帧中int num1[10] = { 1, 2, 3, 4 };//num1为数组名,代表整个数组,都放在栈帧上char char2[] = "abcd";//char2还是一个数组,在栈上  ||  *char2表示第一个元素,该元素在栈上,原理为:常量区有abcd\0字符串,将常量区这一字符串拷贝给栈上的char2数组中,所以在栈上const char* pChar3 = "abcd";//pChar3在栈上  ||  但是这里的*pChar直接就指向了常量区了int* ptr1 = (int*)malloc(sizeof(int) * 4);//ptr1仍然是一个变量,在栈上  ||  *ptr1在堆上,是动态开辟的内存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 在哪里?____//num1为数组名,代表整个数组,都放在栈帧上
char2在哪里?____//char2还是一个数组,在栈上
* char2在哪里?___ //*char2表示第一个元素,该元素在栈上,原理为:常量区有abcd元素,将常量区的abcd拷贝到栈上,所以在栈上
pChar3在哪里?____//pChar3在栈上
* pChar3在哪里?____//但是这里的*pChar直接就指向了常量区了
ptr1在哪里?____//ptr1仍然是一个变量,在栈上
* ptr1在哪里?____//*ptr1在堆上,是动态开辟的内存

2.内存管理方式

2.1C语言内存管理方式

C语言中,使用内存管理的方式为:malloc、realloc、calloc、free

void Test ()
{// 1.malloc/calloc/realloc的区别是什么?int* p2 = (int*)calloc(4, sizeof (int));int* p3 = (int*)realloc(p2, sizeof(int)*10);// 这里需要free(p2)吗?free(p3 );
}
  1. malloc/calloc/realloc的区别?
  2. malloc的实现原理?

链接: malloc的实现原理

2.2C++内存管理方式

C语言的内存管理方式在C++中仍然可以使用,但是在一些情况下会十分麻烦,所以C++就提供了自己的内存管理方式:new和delete

2.2.1new\delete操作内置类型

//new\delete操作内置类型
int main()
{//动态申请一个int类型的空间int* ptr1 = new int;//动态申请一个int类型的空间并初始化为10int* ptr2 = new int(10);//动态申请10个int类型的空间int* ptr3 = new int[10];//动态申请10个int类型的空间并初始化int* ptr4 = new int[10] {1, 2, 3};//1,2,3,0,0,0,0,0,0,0// 使用delete销毁申请的空间delete ptr1;delete ptr2;delete[] ptr3;delete[] ptr4;return 0;
}

2.2.2new\delete操作自定义类型

//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()
{//对于自定义类型,如果使用malloc的话,会显得十分麻烦//而且new/delete 和 malloc/free最大区别是 new/delete对于【自定义类型】除了开空间还会调用构造函数和析构函数A* p1 = (A*)malloc(sizeof(A));A* p2 = new A(1);free(p1);delete p2;A* p3 = (A*)malloc(sizeof(A) * 10);A* p4 = new A[10];free(p3);delete[] p4;//此时如果想要对开辟的类数组初始化:A aa1(1);A aa2(2);A* p6 = new A[10]{ aa1, aa2 };//数组前两个元素使用aa1和aa2初始化,其它元素调用构造函数进行初始化delete[] p6;return 0;
}

还是要提醒一下:在申请自定义类型的空间时,new会调用构造函数,delete会调用析构函数,而malloc和free不会

3.operator new与operator delete函数

这两个函数是系统提供的全局函数,它们底层其实还是调用的malloc和free函数来使用的,new和delete是用户进行动态内存申请和释放的操作符,new在底层调用operator new函数来申请空间,delete在底层调用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)

因为是全局函数,所以这两个函数可以直接调用,与malloc不同的是,仍然会调用构造函数和析构函数:

class A
{
public:A(int a = 0): _a(a){cout << "A():" << this << endl;}~A(){cout << "~A():" << this << endl;}private:int _a;
};int main()
{A* p1 = (A*)operator new(sizeof(A));operator delete(p1);return 0;
}

但是构造函数是不支持这样调用的,析构函数可以这样调用:

class A
{
public:A(int a = 0): _a(a){cout << "A():" << this << endl;}~A(){cout << "~A():" << this << endl;}private:int _a;
};int main()
{A aa1;//直接调用构造函数aa1.A();//err//直接调用析构函数是可行的aa1.~A();return 0;
}

4.new和delete实现原理

4.1内置类型

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

4.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函数来释放空间

5.定位new

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

5.1内存池的基本了解

在这里插入图片描述

5.2定位new的使用方法

定位new的使用格式:

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

使用代码:

class A
{
public:A(int a = 0): _a(a){cout << "A():" << this << endl;}~A(){cout << "~A():" << this << endl;}private:int _a;
};int main()
{A* p1 = (A*)malloc(sizeof(A));//申请一块空间//定位new的意思为:在已分配的原始内存空间中调用构造函数初始化一个对象。new(p1)A;//通过定位new调用构造函数p1->~A();free(p1);A* p2 = (A*)malloc(sizeof(A));//申请一块空间new(p2)A(1);//有参构造函数p2->~A();free(p2);return 0;
}

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

5.3new和delete使用注意事项

new和delete、malloc和free、operator new和operator delete,三者之间不要互相混用,可能会出现报错!

class A
{
public:A(int a = 0): _a(a){cout << "A():" << this << endl;}~A(){delete _pa;cout << "~A():" << this << endl;}private:int _a;int* _pa = new int;
};int main()
{//假设我们使用malloc申请了一块空间A* p1 = (A*)operator new(sizeof(A));delete p1;return 0;
}

如果像上面写的代码一样,就会出现报错!,原因为:
在这里插入图片描述

所以说,不要串联着使用这些关键字,很可能会出错!

正确使用:

class A
{
public:A(int a = 0): _a(a){cout << "A():" << this << endl;}~A(){delete _pa;cout << "~A():" << this << endl;}private:int _a;int* _pa = new int;
};int main()
{A* p1 = (A*)malloc(sizeof(A));new(p1)A;p1->~A();free(p1);A* p2 = (A*)operator new(sizeof(A));new(p2)A(10);p2->~A();operator delete(p2);return 0;
}

6.malloc\free和new\delete的区别

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

相关文章:

C++ 第三讲:内存管理

C 第三讲&#xff1a;内存管理 1.C内存分布2.内存管理方式2.1C语言内存管理方式2.2C内存管理方式2.2.1new\delete操作内置类型2.2.2new\delete操作自定义类型 3.operator new与operator delete函数4.new和delete实现原理4.1内置类型4.2自定义类型 5.定位new5.1内存池的基本了解…...

LeeCode打卡第二十九天

LeeCode打卡第二十九天 第一题&#xff1a;岛屿数量&#xff08;LeeCode第200题&#xff09;: 给你一个由 1&#xff08;陆地&#xff09;和 0&#xff08;水&#xff09;组成的的二维网格&#xff0c;请你计算网格中岛屿的数量。岛屿总是被水包围&#xff0c;并且每座岛屿只…...

阿里云专业翻译api对接

最近我们一个商城项目涉及多语言切换&#xff0c;默认中文。用户切换语言可选英语和阿拉伯语言&#xff0c;前端APP和后端返回动态数据都要根据用户选择语言来展示。前端静态内容都做了三套语言&#xff0c;后端商品为了适用这种多语言我们也进行了改造。每一件商品名称&#x…...

基于Spring Boot的能源管理系统+建筑能耗+建筑能耗监测系统+节能监测系统+能耗监测+建筑能耗监测

介绍 建筑节能监测系统是基于计算机网络、物联网、大数据和数据可视化等多种技术融合形成的一套节能监测系统。 系统实现了对建筑电、水、热&#xff0c;气等能源、资源消耗情况的实时监测和预警、动态分析和评估&#xff0c;为用户建立了科学、系统的节能分析方法&#xff0c…...

大数据新视界 --大数据大厂之 Cassandra 分布式数据库:高可用数据存储的新选择

&#x1f496;&#x1f496;&#x1f496;亲爱的朋友们&#xff0c;热烈欢迎你们来到 青云交的博客&#xff01;能与你们在此邂逅&#xff0c;我满心欢喜&#xff0c;深感无比荣幸。在这个瞬息万变的时代&#xff0c;我们每个人都在苦苦追寻一处能让心灵安然栖息的港湾。而 我的…...

ROS第五梯:ROS+VSCode+C++单步调试

解决问题&#xff1a;在ROS项目中进行断点调试。 第一步&#xff1a;创建一个ROS项目或者打开一个现有的ROS项目。 第二步&#xff1a;修改c_cpp_properties.json 增加一段命令: "compileCommands": "${workspaceFolder}/build/compile_commands.json"第三…...

SLA 概念和计算方法

SLA 概念和计算方法 SLA SLA&#xff1a;服务等级协议&#xff08;简称&#xff1a;SLA&#xff0c;全称&#xff1a;service level agreement&#xff09; 网站服务可用性的一个保证 9越多代表全年服务可用时间越长服务更可靠&#xff0c;停机时间越短&#xff0c;反之亦然…...

C++比大小游戏

目录 开头程序程序的流程图程序游玩的效果下一篇博客要说的东西 开头 大家好&#xff0c;我叫这是我58。 程序 #include <iostream> #include <Windows.h> using namespace std; int main() {int ir 1;char chparr[2] { 0 };int ip1 0;int ip2 0;int i 1;c…...

PCIe进阶之TL:Memory, I/O, and Configuration Request Rules TPH Rules

1 Memory, I/O, and Configuration Request Rules 下述规则适用于 Memory 请求、IO 请求和配置请求。 除了公共的 header 字段外,所有 Memory 请求、IO 请求和配置请求还包括以下字段: (1)Requester ID[15:0] 和 Tag[9:0],组成了 Transaction ID 。 (2)Last DW BE[3:0]…...

【初阶数据结构】一文讲清楚 “堆” 和 “堆排序” -- 树和二叉树(二)(内含TOP-K问题)

文章目录 前言1. 堆1.1 堆的概念1.2 堆的分类 2. 堆的实现2.1 堆的结构体设置2.2 堆的初始化2.3 堆的销毁2.4 添加数据到堆2.4.1 "向上调整"算法 2.5 从堆中删除数据2.5.1 “向下调整”算法 2.6 堆的其它各种方法接口函数 3. 堆排序3.1 堆排序的代码实现 4. TOP-K问题…...

sqli-lab靶场学习(二)——Less8-10(盲注、时间盲注)

Less8 第八关依然是先看一般状态 http://localhost/sqli-labs/Less-8/?id1 然后用单引号闭合&#xff1a; http://localhost/sqli-labs/Less-8/?id1 这关的问题在于报错是不显示&#xff0c;那没办法通过上篇文章的updatexml大法处理。对于这种情况&#xff0c;需要用“盲…...

Dijkstra算法和BFS算法(单源最短路径)

基于你设计的带权有向图&#xff0c;从某一结点出发&#xff0c;执行Dijkstra算法求单源最短路径。用文字描述每一轮执行的过程 文字描述&#xff1a;用BFS算法求单源最短路径的过程 Dijkstra 算法 BFS算法 广度优先算法...

在WordPress中最佳Elementor主题推荐:专家级指南

对于已经在WordPress和Elementor上有丰富经验的用户来说&#xff0c;选择功能强大且高度灵活的主题&#xff0c;能大大提升网站的表现和定制能力。今天&#xff0c;我们来介绍六款适合用户的专家级Elementor主题&#xff1a;Sydney、Blocksy、Rife Free、Customify、Deep和Laye…...

关于RabbitMQ消息丢失的解决方案

RabbitMQ如何保证消息的可靠性传输 一、消息丢失的原因 1. 生产者端 网络问题&#xff1a; 原因&#xff1a;生产者与RabbitMQ服务器之间的网络连接不稳定或中断&#xff0c;导致消息在传输过程中丢失。解决方案&#xff1a;确保网络连接稳定&#xff0c;监控网络状态&#x…...

c语言动态内存分配

前言 我们已经掌握的内存开辟⽅式有&#xff1a; int val 20;//在栈空间上开辟四个字节 char arr[10] {0};//在栈空间上开辟10个字节的连续空间 但是上述的开辟空间的⽅式有两个特点&#xff1a; • 空间开辟⼤⼩是固定的。 • 数组在申明的时候&#xff0c;必须指定数组的…...

零基础制作一个ST-LINK V2 附PCB文件原理图 AD格式

资料下载地址&#xff1a;零基础制作一个ST-LINK V2 附PCB文件原理图 AD格式 ST-LINK/V2是一款可以在线仿真以及下载STM8以及STM32的开发工具。支持所有带SWIM接口的STM8系列单片机;支持所有带JTAG / SWD接口的STM32系列单片机。 基本属性 ST-LINK/V2是ST意法半导体为评估、开…...

nginx基础篇(一)

文章目录 学习链接概图一、Nginx简介1.1 背景介绍名词解释 1.2 常见服务器对比IISTomcatApacheLighttpd其他的服务器 1.3 Nginx的优点(1)速度更快、并发更高(2)配置简单&#xff0c;扩展性强(3)高可靠性(4)热部署(5)成本低、BSD许可证 1.4 Nginx的功能特性及常用功能基本HTTP服…...

监控系列之-Grafana面板展示及制作

一 Grafana设置添加数据源 1、设置Grafana中文显示 最后保存退出&#xff0c;数据源添加完毕 2、导入node_exporter主机监控面板 此处 有外网的情况下&#xff0c;直接输入对应面板的ID号&#xff0c;然后点击加载即可&#xff1b;无无外网的话&#xff0c;则考虑使用上传仪表…...

值传递和地址传递

值传递 我们从下面这段代码开始&#xff1a; point(char*pt); void main(){char b[4]{m,n,o,p},*ptb;point(pt);printf("%c\n",*pt); } point(char *p){p3; }这段代码定义了一个函数 point 和一个主函数 main。 在 main 函数中&#xff0c;定义了一个字符数组 b 并…...

Docker vs. containerd 深度剖析容器运行时

随着容器技术的日益普及&#xff0c;Docker 和 containerd 这两个名词频繁出现在我们的视野中。它们都是容器化技术的重要组成部分&#xff0c;但各自扮演着不同的角色。本文将深入探讨 Docker 和 containerd 的区别与联系&#xff0c;帮助大家更好地理解容器技术的底层原理。 …...

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…...

基础测试工具使用经验

背景 vtune&#xff0c;perf, nsight system等基础测试工具&#xff0c;都是用过的&#xff0c;但是没有记录&#xff0c;都逐渐忘了。所以写这篇博客总结记录一下&#xff0c;只要以后发现新的用法&#xff0c;就记得来编辑补充一下 perf 比较基础的用法&#xff1a; 先改这…...

JAVA后端开发——多租户

数据隔离是多租户系统中的核心概念&#xff0c;确保一个租户&#xff08;在这个系统中可能是一个公司或一个独立的客户&#xff09;的数据对其他租户是不可见的。在 RuoYi 框架&#xff08;您当前项目所使用的基础框架&#xff09;中&#xff0c;这通常是通过在数据表中增加一个…...

管理学院权限管理系统开发总结

文章目录 &#x1f393; 管理学院权限管理系统开发总结 - 现代化Web应用实践之路&#x1f4dd; 项目概述&#x1f3d7;️ 技术架构设计后端技术栈前端技术栈 &#x1f4a1; 核心功能特性1. 用户管理模块2. 权限管理系统3. 统计报表功能4. 用户体验优化 &#x1f5c4;️ 数据库设…...

C++.OpenGL (14/64)多光源(Multiple Lights)

多光源(Multiple Lights) 多光源渲染技术概览 #mermaid-svg-3L5e5gGn76TNh7Lq {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-3L5e5gGn76TNh7Lq .error-icon{fill:#552222;}#mermaid-svg-3L5e5gGn76TNh7Lq .erro…...

基于IDIG-GAN的小样本电机轴承故障诊断

目录 🔍 核心问题 一、IDIG-GAN模型原理 1. 整体架构 2. 核心创新点 (1) ​梯度归一化(Gradient Normalization)​​ (2) ​判别器梯度间隙正则化(Discriminator Gradient Gap Regularization)​​ (3) ​自注意力机制(Self-Attention)​​ 3. 完整损失函数 二…...

Chrome 浏览器前端与客户端双向通信实战

Chrome 前端&#xff08;即页面 JS / Web UI&#xff09;与客户端&#xff08;C 后端&#xff09;的交互机制&#xff0c;是 Chromium 架构中非常核心的一环。下面我将按常见场景&#xff0c;从通道、流程、技术栈几个角度做一套完整的分析&#xff0c;特别适合你这种在分析和改…...

DeepSeek源码深度解析 × 华为仓颉语言编程精粹——从MoE架构到全场景开发生态

前言 在人工智能技术飞速发展的今天&#xff0c;深度学习与大模型技术已成为推动行业变革的核心驱动力&#xff0c;而高效、灵活的开发工具与编程语言则为技术创新提供了重要支撑。本书以两大前沿技术领域为核心&#xff0c;系统性地呈现了两部深度技术著作的精华&#xff1a;…...

leetcode73-矩阵置零

leetcode 73 思路 记录 0 元素的位置&#xff1a;遍历整个矩阵&#xff0c;找出所有值为 0 的元素&#xff0c;并将它们的坐标记录在数组zeroPosition中置零操作&#xff1a;遍历记录的所有 0 元素位置&#xff0c;将每个位置对应的行和列的所有元素置为 0 具体步骤 初始化…...

门静脉高压——表现

一、门静脉高压表现 00:01 1. 门静脉构成 00:13 组成结构&#xff1a;由肠系膜上静脉和脾静脉汇合构成&#xff0c;是肝脏血液供应的主要来源。淤血后果&#xff1a;门静脉淤血会同时导致脾静脉和肠系膜上静脉淤血&#xff0c;引发后续系列症状。 2. 脾大和脾功能亢进 00:46 …...