malloc(字节) ————申请 calloc(数量,每个字节大小)————申请并初始化为0 realloc ()扩展或者压缩 realloc 扩展可能导致位置改变,所以要重新接收,压缩会截尾 free()释放,
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接口释放掉;被释放的空间不能重复释放;
在这里 我们发现 栈是向下生长的(栈顶是高地址);堆是向上生长的
注意 :动态分配内存是在堆上进行的
操作符 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(指针)+类型名+(初始化列表)
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的用法;