1 C
malloc(字节) ————申请
calloc(数量,每个字节大小)————申请并初始化为0
realloc ()扩展或者压缩
realloc 扩展可能导致位置改变,所以要重新接收,压缩会截尾
free()释放,
C 详细
malloc 接口 申请 free释放
int arrary= (int*) malloc(nsizeof( int));malloc 的返回值 为 void;开辟
失败,返回NULL;
free(NULL 或者malloc 的返回指针)
free的操作注意点 : 不能部分释放 ;申请多少,释放多少,编译可以通过,但运行一点出错;free(NULL)永真;
2 calloc接口, 申请时并初始化0;calloc(元素数量,每个元素的字节数)
realloc :
对已申请的内存做修改:扩展和缩小
扩展:原先内容仍旧保留,
缩小:去尾;
int* array1= realloc(array被操作数组,调整后的大小);
注意 :realloc 函数的地址可能会变 :
因为:内存(堆)不够大时,编译器就会去寻找 合适 的内存大小的地方 去开辟;所以地址可能 会改变;
若是改变,原来的指针(申请的内存块)将被realloc接口释放掉;被释放的空间不能重复释放;
在这里 我们发现 栈是向下生长的(栈顶是高地址);堆是向上生长的
注意 :动态分配内存是在堆上进行的
2 C++
操作符
new 创建(申请)
delete 释放
new 类型[ ]
delete[ ] 指针
1 C++的内存管理方式:
void Test()
{
//动态申请一个int 类型空间
int* ptr=new int;
//申请并初始化
int* ptr1 =new int(3);
//动态申请连续空间
int* ptr2 =new int[10];
//释放
delete ptr;
delete ptr1;
delete[ ] ptr2;
}
2 new 和 delete 在申请自定义类型空间时,new会调用构造函数, delete 会调用析构函数
new 实际调用 operater new 再底层调用 malloc,如果申请成功,返回一个只想申请空间地址的指针,申请失败会查看应对措施,成功返回指针,失败抛出异常
delete 底层调用 free 释放空间
operate new 和 operate delete 类的专属重载
一般只用来在类中检查内存泄漏 ,定位内存泄露的位置
类 的new 和 delete 的原理
new调用malloc 在申请空间上执行类的构造函数,
delete 调用类的析构函数,先清理类中的资源,再调用free释放申请的对象的空间
new [] 和 delete []实际上是执行多次;
定位new表达式 new placement ;
这个表达式用于在已知申请的空间上创建对象,一般配合内存池使用,因为内存池申请的空间没有初始化(我们实际上经常用空间配置器 allocator)
定位new表达式格式
new+(指针)+ 类型名 或new(指针)+类型名+(初始化列表)
mallo /free 和 new /delete 的区别:
1 malloc 和free 是函数,new delete 是操作符;
2 malloc 申请的空间没有初始化,new 可以初始化
3 malloc 申请空间时,需要手动计算所需空间大小,new会自己计算,我们只需要接上类型就可;
4 malloc 返回值为 void* 我们需要时必须强转,new不需要,new后接类型
5 malloc 申请失败返回 NULL;,因此使用是必须判空,new需要捕获异常;
6 申请自定义类型对象空间时,malloc 只开辟空间,new 会调用构造函数构造出一个对象,free 不会调用析构函数。而delete 会调用析构函数释放内中的资源,再调用free空间
7malloc free是在堆上 操作,而new delete 不一定在堆上 (例如类中申请对象时必须在栈上,空间适配器allocator等)
new的空间不用delete的例子:
new placement/内存池:
如果我们使用非堆上的空间则不要delete;这时new只是在已经存在的空间上执行一个构造函数,不管理空间分配,因而空间的管理权不为new对象所有,new对象只要在最后保证执行了析构函数就可以了----new placement的用法;