使用定位new运算符需要注意的问题

什么是定位new运算符:为什么使用定位new运算符
在指定位置创建对象;通过特定空间地址访问硬件;分配内存时制定内存位置。
在这里插入图片描述

//cpp
#include <new>
const int BUF = 512;
char buffer[BUF]; //定义静态空间为new运算符提供内存空间

double* ptr = new (buffer) double[10]; 
//此时创建的double数组放在静态空间buffer中
std::cout<<"address:"<<(void*)buffer<<"same as "<<ptr<<std::endl;

注:buffer强制类型转换为void*原因:若不转换,则输出字符串内容。
1、定位new运算符接受两个参数:地址 与 所需空间大小。
2、定位new运算符不考虑指向地址是否已有值,直接使用。将此部分考虑内容转嫁给程序员。
3、此部分内存不可用delete ptr 释放,因此是静态内存,不受动态内存delete管辖。 静态内存不能释放,随程序结束自动释放。

与对象结合使用时需注意的问题

问题定位:对象使用了定位new创建与析构函数delete的冲突。
delete可与常规new运算符配合使用,但不能与定位new运算符配合使用。

const int BUFF = 50;
char * buffer = new char[50];

classA * ptr1 = new (buffer) classA;
classA * ptr2 = new (buffer+sizeof(classA)) classA;
delete ptr2; //此处引发运行时错误,原因见注脚1。
delete ptr1; //此处释放错误,见注脚2。
delete [] buffer; //此时释放了内存,却没有调用classA的析构函数 ,
//见注脚3。

注:
1、ptr2定位new的指针,与delete不匹配,因此引发运行时错误。
2、ptr1=buffer(两地址相同),首先deleteptr1定位new不匹配,因此不释放ptr1指向的classA对象;
其次由于其等于buffer因此释放buffer位置一个空间。由于buffer为 new[] 申请,应该用delete [] 释放。
3、未通过定位new销毁对象,但空间已经不存在。解决方法:显示的调用对象析构函数(属于少见情况)

ptr2->~classA (); //需要注意销毁顺序

不抛出异常的new:

int * point = new (std::nothrow) int;
if(point ==0){...} //分配失败则。。。
// 禁止内存分配错误输出异常,而是返回NULL指针。 

在1993以前,operator new 在失败时返回null,之后规定返回bad_alloc异常。为了旧代码(如以前的异常检测:point==0)仍然可用,C++标准委员会提供了nothrow版本的new,如上所示。使得分配失败不抛出异常而是继续返回null
注:nothrow有局限性,只适合内存分配,而后续构造函数若使用new仍可能抛出异常。

总结:

1、定位new运算符与delete不匹配,使用引发运行时错误。
2、获取字符串指针的地址,需要强制类型转换为void*
3、定位new运算符创建的对象,不可用delete classAptr,需要显示调用对象析构函数。
4、new (std::nothrow)不抛出异常。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值