malloc与free是C++/C语言的标准库函数。分别负责分配内存和释放内存。
new/delete是C++的运算符。分别负责分配内存和调用构造函数 以及 调用析构函数和释放内存;
由于malloc/free是库函数而不是运算符,不在编译器控制权限之内,不能够把执行构造函数和析构函数的任务强加于malloc/free。因此C++语言需要一个能完成动态内存分配和初始化工作的运算符new,以及一个能完成清理与释放内存工作的运算符delete。注意new/delete不是库函数
delete只会调用一次析构函数,而delete[]会调用每一个成员的析构函数。
在More Effective C++中有更为详细的解释:“当delete操作符用于数组时,它为每个数组元素调用析构函数,然后调用operator delete来释放内存。”
delete与new配套,delete []与new []配套
补充:delete []是如何得知数组个数的
转载出处:https://blog.csdn.net/lcfeng1982/article/details/8106719
通常在delete[]一个数组时,编译器要按顺序作如下2件事情:
1. 对每个数组元素调用析构函数 。
2. 释放对象数组所占内存 。编译器只需简单地把数组首地址告诉操作系统,操作系统内部有内存申请情况的记录(每个申请内存段的首地址,长度,etc..),因此会正确地释放掉内存.注意整个数组所占内存是一次释放掉的,而不是每个元素释放一次.事实上,在这一步编译器无需知道数组元素个数.以上2步是彼此独立的2步. 另外,如果数组元素没有显式析构函数(例如char),那么编译器就无需作第1步了.在这种情况下,编译器根本不需要知道数组个数,因此就偷懒不再在数组前面放元素个数n了.
{
public:
TestA(){a=new int(1);}
~TestA(){delete a;}
int* a;
};
memset(pInt, 0, 100*sizeof(int));
int* pr = pInt-1;
int size = *pr;//此处size的值不是100,说明数组前面没有加上数组个数(因为int不需要显示析构函数)
delete []pInt;
TestA* pArT = new TestA[100];
size = sizeof(TestA) * 100;
int* pp = (int*)(pArT) - 1;
size = *pp;//此处size的值是100,说明数组前面加上了数组个数
delete []pArT;
在第1步确实需要知道元素个数.编译器会把元素个数放在分配的那块内存的前面,结构形如 :n object1 object2 ... objectn 。另外,不要把这个n和数组内存长度混淆起来。前者是编译器管理的,后者是操作系统管理的.(如果知道对象的size,是可以从后者计算出前者.遗憾的是,操作系统没有API提供所申请的内存段的长度,编译器只好自己记录)
另外,提醒C++程序员 delete [] p;语句中delete和[]之间一定要有空格
举例如下:
class TestA
int *pInt = new int[100];