浅析new与delete工作原理及注意事项

17 篇文章 0 订阅

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对象,第一个调用
//了析构函数,后两个没有,所以后两个对象内含的指针数据成员所指内存泄露)
  1. 只要在new后使用了delete(不论是否是array new),那么对象所分配的内存都会释放,但是如果new的是array,而使用的是delete(应该使用delete[]),且对象内含有指针,那么指针这个对象所分配的内存被释放(一般是4字节),但是指针所指向的内存不会被释放。
  2. 补充:在创建类时可根据类内是否含有指针数据成员分为两大类,若没有,一般可以不用写big three(five)即拷贝构造、拷贝赋值、析构函数(移动构造、移动赋值),但是若含有指针数据成员,一般必须写big three(five)。
  3. 比如Complex类仅含实部虚部两个double型数据成员,使用默认的big three(five)即可,而String类含有char*类数据成员,必须自己写big three(five),默认的只是浅拷贝(拷贝的只是指针的地址),而显然我们需要的是深拷贝(拷贝指针指向的地址的内容)。
  4. 结合1与3所述,对Complex类(不含指针数据成员)array new 后使用delete或array delete效果相同,但是对于String类(含有指针),array new 后必须使用array delete(当然,不要犟着说new的array只用一个元素)。
    综上所述,array new 必须要搭配array delete 使用,一定不会出错
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值