C\C++内存管理(未完结)

一.C\C++内存分布

C和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在哪里?____       
pChar3在哪里?____      
ptr1在哪里?____        
2. 填空题:
sizeof(num1) = ____;   
*char2在哪里?___
 *pChar3在哪里?____
 *ptr1在哪里?____
 sizeof(char2) = ____;      
sizeof(pChar3) = ____;     
sizeof(ptr1) = ____;
 3. sizeof 和 strlen 区别?
strlen(char2) = ____;
 strlen(pChar3) = ____;

答案
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
【说明】

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

二.C语言中动态内存管理方式:malloc/calloc/realloc/free

void Test ()
 {
 int* p1 = (int*) malloc(sizeof(int));
 free(p1);
 // 1.malloc/calloc/realloc的区别是什么?
int* p2 = (int*)calloc(4, sizeof (int));
 int* p3 = (int*)realloc(p2, sizeof(int)*10);
 // 这里需要free(p2)吗? 不需要 原地扩容p3 还是p2接收地址 异地扩容 自动free(p2)
free(p3 );
 }

【面试题】

  1. malloc/calloc/realloc的区别?
  2. malloc的实现原理? glibc中malloc实现原理

三.C++内存管理方式

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

3.1.new/delete操作内置类型

void Test()
 {
 // 动态申请一个int类型的空间
int* ptr4 = new int;
 // 动态申请一个int类型的空间并初始化为10
 int* ptr5 = new int(10);
 // 动态申请10个int类型的空间
int* ptr6 = new int[3];
 delete ptr4;
 delete ptr5;
 delete[] ptr6;
 }
int main()
{	
	//开辟一个空间
	int* ptr = new int;//开辟一共int类型的空间 在堆区 返回一共堆上开辟好的空间地址

	//开辟一个空间并且初始化
	int* ptr1 = new int(10); //只需要在开辟空间类型中+(初始化值)即可
	cout <<"ptr1存放地址对应值:" << *ptr1 << endl;

	//开辟多个空间
	int* ptr2 = new int[10];//在开辟空间类型后加上+[开辟个数]即可 并返回开辟数组(多个空间)的首地址
	
	//开辟多个空间并且初始化 在之前c++是不允许开辟的数组初始化的 C++11就可以了
	int* ptr3 = new int[10] {1, 2, 3, 4, 5};//开辟数组空间 初始化用{里面写初始化值} 可以完全初始化 也可不完全初始化 不完全初始化的空间默认初始化为0
	int arr[5] = { 1,2,3,4,5 };

	//将开辟好的空间 释放掉 
	delete(ptr);
	delete(ptr1);
	delete[](ptr2); //数组的释放 连续的空间释放+delete[] 指针存放地址
	delete[](ptr3);




	return 0;
}

在这里插入图片描述
注意:申请和释放单个元素的空间,使用new和delete操作符,申请和释放连续的空间,使用new[]和delete[],注意:匹配起来使用。

3.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)); // C
 int* 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不会。
new开辟空间失败会抛异常,不需要手动检查
抛异常 捕获异常
在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

四.operator new与operator delete函数(重要点进行讲解)

4.1. operator new与operator delete函数(重点)

五.new和delete的实现原理

5.1.内置类型

5.2 自定义类型

六. 定位new表达式(placement-new) (了解)

七. 7. 常见面试题

7.1. malloc/free和new/delete的区别

7.2 内存泄漏

  • 16
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值