C++ 第三讲:内存管理

news2024/11/15 10:24:27

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类型的空间并初始化为10
	int* 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 bytes
	void* 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_FINALLY
		return;
}

/*
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在释放空间前会调用析构函数完成
  空间中资源的清理释放

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

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

相关文章

基于Java的学生档案管理系统的设计与实现

基于springbootvue实现的学生档案管理系统 &#xff08;源码L文ppt&#xff09;4-065 第4章 系统设计 4.1 总体功能设计 学生档案管理系统的总体功能设计包括学生信息管理、课程管理、教师信息管理、成绩管理和系统配置管理。系统将提供用户友好的界面&#xff0c;支…

【网络】传输层协议TCP

TCP协议 TCP&#xff08;Transmission Control Protocol&#xff0c;传输控制协议&#xff09;是一种面向连接的、可靠的、基于字节流的传输层通信协议&#xff0c;由IETF的RFC 793定义。TCP在IP&#xff08;Internet Protocol&#xff0c;互联网协议&#xff09;网络层上提供…

【Linux】—— muduo网络库的安装配置与使用

muduo网络库编程 Linux环境下Epollpthread线程库 Boost库安装与使用 安装Boost库 下载boost库源码&#xff0c;linux环境解压 tar -zxvf boost_1_69_0.tar.gz 解压完成后&#xff0c;进入该目录&#xff0c;查看内容 运行bootstrap.sh工程编译构建程序 ./bootstrap.sh …

javascript-代码执行原理

js 是解释型语言 js 引擎执行流程 分为两个阶段: 语法分析执行阶段执行阶段涉及的数据结构: 调用栈。处理执行上下文和执行代码内存堆。给对象分配内存任务队列。暂存待执行的任务,分为宏任务队列和微任务队列语法分析 词法分析 > 语法分析 > 代码生成(字节码) …

封装svg图片

前言 项目中有大量svg图片&#xff0c;为了方便引入&#xff0c;所以对svg进行了处理 一、svg是什么&#xff1f; svg是可缩放矢量图形&#xff0c;是一种图片格式 二、使用步骤 1.创建icons文件夹 将icons文件夹放进src中&#xff0c;并创建一个svg文件夹和index.js&…

深入探索迭代器模式的原理与应用

&#x1f3af; 设计模式专栏&#xff0c;持续更新中 欢迎订阅&#xff1a;JAVA实现设计模式 &#x1f6e0;️ 希望小伙伴们一键三连&#xff0c;有问题私信都会回复&#xff0c;或者在评论区直接发言 迭代器模式 &#x1f4bb; 迭代器模式 (Iterator Pattern) 是一种行为设计模…

【LeetCode】每日一题 2024_9_19 最长的字母序连续子字符串的长度(字符串,双指针)

前言 每天和你一起刷 LeetCode 每日一题~ LeetCode 启动&#xff01; 题目&#xff1a;坐上公交的最晚时间 代码与解题思路 func longestContinuousSubstring(s string) (ans int) { // 题目要求&#xff1a; 最长 的 字母序连续子字符串 的长度// 双指针&#xff0c;start …

【学习笔记】线段树分裂

前言 有线段树合并就应该有线段树分裂。它是线段树合并的逆过程。具体的&#xff0c;你需要以权值线段树中第 k 小的数为分界线&#xff0c;把线段树分成两半。 算法流程 和线段树上二分类似。假设原来的线段树为 u&#xff0c;要分裂出线段树 v 记左子树的权值为 val。如果…

CodeMeter助力软件授权与IP保护,保障工业自动化与物联网安全

随着工业自动化的飞速发展&#xff0c;Hilscher的开放工业4.0联盟&#xff08;OI4&#xff09;旗舰店应运而生&#xff0c;将应用商店模式引入工业领域。凭借CodeMeter授权和加密技术的支持&#xff0c;该商店为工业用户提供了一个安全且开放的应用程序和解决方案平台。该平台不…

超声波清洗机哪个品牌好用又实惠?精选业内四款优质清洗机推荐

超声波清洗机作为一种创新的清洁解决方案&#xff0c;凭借其深入微观的清洁效能、简便的操作方式以及对物品的细腻呵护&#xff0c;正逐渐成为广受喜爱的清洁良品。不过&#xff0c;市面上品牌林立、型号多样&#xff0c;价格亦波动不一&#xff0c;这无疑为消费者选购时平添了…

ATT&CK靶机实战系列之vulnstack2

声明: 本文章只是用于网络安全交流与学习&#xff0c;若学者用学到的东西做一些与网络安全不相关的事情&#xff0c;结果均与本人无关&#xff01;&#xff01;&#xff01; 靶场环境: 使用kali作为hacker的攻击机器&#xff0c;来对web pc dc进行攻击。 这里声明一下: 关于…

Cesium billboard 自定义shader实现描边效果

Cesium billboard 自定义shader实现描边效果 uniform sampler2D u_atlas;uniform vec2 dimensions;in vec2 v_textureCoordinates;in vec4 v_pickColor;in vec4 v_color;in float v_splitDirection;void main(){if (v_splitDirection < 0.0 && gl_FragCoord.x > …

一文详解可视化大屏技术在地震监测中的作用!

昨天&#xff0c;安徽合肥市肥东县发生地震&#xff0c;震级达到4.7级&#xff0c;震源深度12千米。这一事件再次提醒我们&#xff0c;地震的威胁无处不在&#xff0c;及时有效的地震预警对于减少灾害损失至关重要。四川省作为地震活动频繁的地区&#xff0c;近年来在地震监测和…

配置环境-keil

配置keil -- 先将keil安装配置好&#xff0c;包括库 一、STM32 -- STM32是意法半导体&#xff08;意大利&#xff09;采用ARM公司设计的内核&#xff0c;设计一系列32位单片机芯片。 1、STM32开发的几种方式 2、STM32寄存器和库函数版本的工程创建 新建文件夹 复制相关文件…

【机器学习(八)】分类和回归任务-因子分解机(Factorization Machines,FM)-Sentosa_DSML社区版

文章目录 一、算法概念二、算法原理&#xff08;一&#xff09; FM表达式&#xff08;二&#xff09;时间复杂度&#xff08;三&#xff09;回归和分类 三、算法优缺点&#xff08;一&#xff09;优点&#xff08;二&#xff09;缺点 四、FM分类任务实现对比&#xff08;一&…

【2024华为杯研究生数学建模竞赛】比赛思路、代码、论文更新中.....

目录 赛中助攻华为杯常用建模算法&#x1f5d2;️&#x1f5d2;️历年优秀论文⭐⭐论文模板1&#xff09;论文模板2&#xff09;基础画图能力 绘图与数据分析软件SPSSPRO 2024研究生数学建模竞赛时间为9月21日&#xff08;周六&#xff09;8:00至9月25日&#xff08;周三&#…

互联网前端之 CSS 常见属性与三层结构

目录 现在互联网前端三层 CSS 常见属性 关注作者微信公众号&#xff0c;开启探索更多 CSS 知识的精彩之旅。在这里&#xff0c;你将收获丰富的CSS专业内容&#xff0c;深入了解这一网页开发语言的奥秘&#xff0c;不断拓展你的知识边界&#xff0c;提升技能水平。快来关注吧&…

对想学习人工智能或者大模型技术从业者的建议,零基础入门到精通,收藏这一篇就够了

“ 技术的价值在于应用&#xff0c;理论与实践相结合才能事半功倍**”** 写这个关于AI技术的公众号也有差不多五个月的时间了&#xff0c;最近一段时间基本上都在保持日更状态&#xff0c;而且写的大部分都是关于大模型技术理论和技术方面的东西。‍‍‍‍‍‍‍‍‍ 然后最近…

Windows安全日志分析(事件ID详解)

目录 如何查看Windows安全日志 常见事件ID列表 事件ID 1116 - 防病毒软件检测到恶意软件 事件ID 4624 - 账户登录成功 事件ID 4625 - 账户登录失败 事件ID 4672 - 为新登录分配特殊权限 事件ID 4688 - 新进程创建 事件ID 4689 - 进程终止 事件ID 4720 - 用户账户创建 …

3款免费的GPT类工具

前言 随着科技的飞速发展&#xff0c;人工智能&#xff08;AI&#xff09;的崛起与发展已经成为我们生活中不可或缺的一部分。它的出现彻底改变了我们与世界互动的方式&#xff0c;并为各行各业带来了前所未有的便利。 一、Kimi 网址&#xff1a;点我前往 国产AI模型Kimi是一…