C和C++的内存管理
1 C++内存分布
程序中需要存储一些数据:局部数据(存在栈帧)、静态数据和全局数据(存在静态区/数据段)、常量数据(存在常量区/代码段)、动态申请数据(存在堆上)
建立栈帧本质上是为了即用即销毁——存储局部数据
1. 栈又叫堆栈--非静态局部变量/函数参数/返回值等等,栈是向下增长的。
2. 内存映射段是高效的I/O映射方式,用于装载一个共享的动态内存库。用户可使用系统接口
创建共享共享内存,做进程间通信。(Linux课程如果没学到这块,现在只需要了解一下)
3. 堆用于程序运行时动态内存分配,堆是可以上增长的。
4. 数据段--存储全局数据和静态数据。
5. 代码段--可执行的代码/只读常量
堆区数组中保存的数据也是开辟在堆区内存的,在释放资源时,要把堆内存数组中的数据逐个释放
2 C语言中动态内存管理方式
malloc、calloc、realloc和free
3 C++的内存管理方式
C
语言内存管理方式在
C++
中可以继续使用,但有些地方就无能为力,而且使用起来比较麻烦,因
此
C++
又提出了自己的内存管理方式:
通过
new
和
delete
操作符进行动态内存管理
。
区别:malloc失败返回空指针,new失败抛异常
3.1 new和delete
![](https://i-blog.csdnimg.cn/blog_migrate/5b8af12a300f1dbca7c6b5436659a07a.png)
不需要计算新开辟的空间需要多少个字节,也不需要强制类型转换
申请和释放单个元素的空间,使用new和delete操作符
申请和释放连续的空间,使用new[]和delete[ ]操作符
![](https://i-blog.csdnimg.cn/blog_migrate/f9306e9a6820cd8105fa6dd27e1f0d36.png)
这里对于内置类型的区别不大,主要是自定义类型,期望对象定义出来的时候就能调用构造函数初始化
自定义类型:C语言是单纯地开空间,C++是开空间+调用构造函数初始化,delete会自动调用析构函数
new+类型 delete+指针。
内置类型一定要匹配使用
3.2 new和delete操作自定义类型
![](https://i-blog.csdnimg.cn/blog_migrate/563048bd52371b21049a7bdd5f575e3e.png)
在申请和释放自定义类型空间时,new是开辟空间+调用构造函数初始化;
delete是调用析构函数+释放空间。
而malloc和free只会开辟空间和释放空间,而不能去调用构造函数和析构函数。
4 operate new 和operate delete
operator new和operator delete不是操作符重载,而是函数。
![](https://i-blog.csdnimg.cn/blog_migrate/0adc3f16beefa0e9cd102a0ed3eb4b93.png)
![](https://i-blog.csdnimg.cn/blog_migrate/1c03c53d0333fd6392514df48e183531.png)
operator new和malloc的功能一样,但是不同的是operator失败后抛异常,而malloc失败返回空指针
operator delete最终是通过free来释放空间的
new和delete的底层
new = 先调用operate new,再调用构造函数
delete = 先调用析构函数,再调用oprate delete
显式调用构造函数初始化
如果我们想要使用自己实现的类专属的operator new / operator delete,那么就要先调用专属的函数,然后再进行显式调用构造函数去初始化
![](https://i-blog.csdnimg.cn/blog_migrate/c7bd4a950e4a0d587867190b14b1cccd.png)
5 new和delete实现原理
5.1 new原理
①调用operator new函数申请空间
②在申请的空间上执行构造函数,完成对象的构造
5.2 delete原理
①在空间上执行析构函数,完成对中资源的清理工作
②调用operator delete函数释放对象的空间
5.3 new T[N]原理
①调用operator new[]函数,在operator new[]中实际调用operator new函数完成N个对象空间的申请
②在申请的空间上执行N次构造函数
5.4 delete[ ]原理
①在释放的对象空间上执行N次析构函数,完成N个对象中资源的清理
②调用operator delete[]释放空间,实际在operator delete[]中调用operator delete来释放空间