定位new运算符的使用

最近在研究STL的空间配置器的时候,使用到内存池和定位new的场景,感觉知识优点遗忘,总结总结
首先,什么是定位new??
官方解释:
《定位new运算符的使用》
我的理解:先申请一块内存空间,然后在这段空间上来构造对象(确定对象的位置),常使用到的场景是内存池
内置类型使用规则:

#include <new>//或者使用#include<iostream> ,因为<iostream>中包含<new>
char buffer[1024]; //定义静态空间为new运算符提供内存空间
double* ptr = new (buffer) double[10]; 
//此时创建的double数组放在栈空间中
std::cout<<"address:"<<(void*)buffer<<"same as "<<ptr<<std::endl;

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

char *buffer=new char[1024]; //定义动态空间为new运算符提供内存空间
double* ptr = new (buffer) double[10]; 
delete[] buffer;

自定义类型使用:

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

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

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

    原文作者:梦T醒
    原文地址: https://blog.51cto.com/14233078/2492019
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞