一、C语言动态内存管理方式
在C语言中,动态内存管理的接口有malloc/calloc/realloc,那么三者之间有什么区别呢?
简单来说就是malloc是动态开辟空间;calloc是开辟空间的同时按字节进行初始化为0,并且可以扩容,须知道要扩容到多大;realloc是对当前空间进行增容,对已有空间进行操作,空间足够大时原地扩容,否则另外扩容。
详细可见【C语言】malloc,calloc,realloc的区别
二、C++动态内存管理方式
C语言内存管理方式在C++中可以继续使用,但有些地方就无能为力而且使用起来比较麻烦,因此C++又提出了自己的内存管理方式:通过new和delete操作符进行动态内存管理。
new/delete操作内置类型
先来看段代码:
void Test()
{
// 动态申请一个int类型的空间
int* ptr4 = new int;
// 动态申请一个int类型的空间并初始化为10
int* ptr5 = new int(10);
// 动态申请10个int类型的空间
int* ptr6 = new int[10];
delete ptr4;
delete ptr5;
delete[] ptr6;
}
【总结】:申请和释放单个元素的空间,使用new和delete操作符,申请和释放连续的空间,使用new[]和delete[]
new/delete操作自定义类型
#include <iostream>
using namespace std;
class Date
{
public:
Date(int year=1990, int month=1, int day=1)
:_year(year)
, _month(month)
, _day(day)
{}
~Date()
{
cout << "~Date()" << endl;
}
private:
int _year;
int _month;
int _day;
};
int main()
{
//基本类型(内置类型)
int* p1 = new int;//C++开辟
int* p2 = (int*)malloc(sizeof(int));//C语言开辟
delete p1;
free(p2);
//自定义类型
Date* p3 = new Date;//开空间+初始化
Date* p3 = new Date(2019,3,2);//开空间+初始化
Date* p4 = (Date*)malloc(sizeof(Date));//开空间
delete p3;//析构(清理)+释放空间
delete p4;//析构(清理)+释放空间
free(p4);//释放空间
system("pause");
return 0;
}
【总结】:在申请自定义类型的空间时,new会调用构造函数,delete会调用析构函数,而malloc与free不会。一定要匹配使用,不要混用!!!
三、operator new与operator delete函数
new和delete是用户进行动态内存申请和释放的操作符,operator new 和operator delete是系统提供的全局函数,new在底层调用operator new全局函数来申请空间,delete在底层通过operator delete全局函数来释放空间。用法如下:
int* p1 = (int*)operator new(sizeof(int));
operator delete(p1);
我们可以看出operator new 、operator delete 与 malloc 、free 的用法完全一样,那么他们之间的区别是什么呢?
它们主要在于处理错误的方式有区别:operator new失败抛异常,而malloc失败返回NULL(0)
malloc/free和new/delete的区别
malloc/free和new/delete的共同点是:都是从堆上申请空间,并且需要用户手动释放。不同的地方是:
- malloc和free是函数,new和delete是操作符
- malloc申请的空间不会初始化,new可以初始化
- malloc申请空间时,需要手动计算空间大小并传递,new只需在其后跟上空间的类型即可
- malloc的返回值为void*, 在使用时必须强转,new不需要,因为new后跟的是空间的类型
- malloc申请空间失败时,返回的是NULL,因此使用时必须判空,new不需要,但是new需要捕获异常
- 申请自定义类型对象时,malloc/free只会开辟空间,不会调用构造函数与析构函数,而new在申请空间后会调用构造函数完成对象的初始化,delete在释放空间前会调用析构函数完成空间中资源的清理
- new/delete比malloc和free的效率稍微低点,因为new/delete的底层封装了malloc/free