文章目录
- 一.C\C++内存分布
- 二.C语言中动态内存管理方式:malloc/calloc/realloc/free
- 三.C++内存管理方式
- 3.1.new/delete操作内置类型
- 3.2.new和delete操作自定义类型
- 四.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 内存泄漏
一.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) = ____;
答案
【说明】
- 栈又叫堆栈–非静态局部变量/函数参数/返回值等等,栈是向下增长的。
- 内存映射段是高效的I/O映射方式,用于装载一个共享的动态内存库。用户可使用系统接口
创建共享共享内存,做进程间通信。(Linux课程如果没学到这块,现在只需要了解一下) - 堆用于程序运行时动态内存分配,堆是可以上增长的。
- 数据段–存储全局数据和静态数据。
- 代码段–可执行的代码/只读常量。
二.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 );}
【面试题】
- malloc/calloc/realloc的区别?
- malloc的实现原理? glibc中malloc实现原理
三.C++内存管理方式
C语言内存管理方式在C++中可以继续使用,但有些地方就无能为力,而且使用起来比较麻烦,因
此C++又提出了自己的内存管理方式:通过new和delete操作符进行动态内存管理。
3.1.new/delete操作内置类型
void Test(){// 动态申请一个int类型的空间
int* ptr4 = new int;// 动态申请一个int类型的空间并初始化为10int* 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};//开辟数组空间 初始化用{里面写初始化值} 可以完全初始化 也可不完全初始化 不完全初始化的空间默认初始化为0int 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)); // 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不会。
new开辟空间失败会抛异常,不需要手动检查
抛异常 捕获异常