什么是定位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
(两地址相同),首先delete
与ptr1定位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)
不抛出异常。