new:先分配内存再调用构造函数
当我们写出
Complex* pc=new Complex(1,2);//Complex是一个复数类,此时实部为1,虚部为2
编译器转化为
void* men=operator new(sizeof(Complex));//分配内存,operator new是一个特殊的函数,内部调用malloc(n)
pc=static_cast<Complex*>(mem);//转型
pc->Complex::Complex(1,2);//构造函数
delete:先调用析构函数,再释放内存
delete ps;//ps是有指针成员函数的类的指针(这里是自己写的string)
编译器转化为
String::~String(ps);//析构函数
operator delete(ps);//释放内存(operator delete是一个特有函数,内部调用free(ps))
array new一定要搭配array delete使用
正确写法:
String* ps=new String[3];//自己写的String类内含指针数据成员
...
delete[] p;//调用三次析构函数,很正确
错误写法:
String* ps=new String[3];
...
delete p;//调用一次析构函数,造成内存泄露(数组中3个String对象,第一个调用
//了析构函数,后两个没有,所以后两个对象内含的指针数据成员所指内存泄露)
- 只要在new后使用了delete(不论是否是array new),那么对象所分配的内存都会释放,但是如果new的是array,而使用的是delete(应该使用delete[]),且对象内含有指针,那么指针这个对象所分配的内存被释放(一般是4字节),但是指针所指向的内存不会被释放。
- 补充:在创建类时可根据类内是否含有指针数据成员分为两大类,若没有,一般可以不用写big three(five)即拷贝构造、拷贝赋值、析构函数(移动构造、移动赋值),但是若含有指针数据成员,一般必须写big three(five)。
- 比如Complex类仅含实部虚部两个double型数据成员,使用默认的big three(five)即可,而String类含有char*类数据成员,必须自己写big three(five),默认的只是浅拷贝(拷贝的只是指针的地址),而显然我们需要的是深拷贝(拷贝指针指向的地址的内容)。
- 结合1与3所述,对Complex类(不含指针数据成员)array new 后使用delete或array delete效果相同,但是对于String类(含有指针),array new 后必须使用array delete(当然,不要犟着说new的array只用一个元素)。
综上所述,array new 必须要搭配array delete 使用,一定不会出错