博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
C++中new的三种使用方法说明
阅读量:6371 次
发布时间:2019-06-23

本文共 2711 字,大约阅读时间需要 9 分钟。

一、概念

在C++中new的三种用法包括:plain new, nothrow new 和 placement new。

plain new 就是我们最常使用的new的方式,在C++中的定义如下:

1 void* operator new(std::size_t) throw(std::bad_alloc);  2 void operator delete( void *) throw();

plain new在分配失败的情况下,抛出异常std::bad_alloc而不是返回NULL,因此通过判断返回值是否为NULL是徒劳的。

nothrow new 是不抛出异常的运算符new的形式。nothrow new在失败时,返回NULL。定义如下:

1 void * operator new(std::size_t, const std::nothrow_t&) throw();2 void operator delete(void*) throw();

placement new 意即“放置”,这种new允许在一块已经分配成功的内存上重新构造对象或对象数组。placement new不用担心内存分配失败,因为它根本不分配内存,它做的唯一一件事情就是调用对象的构造函数。定义如下:

1 void* operator new(size_t, void*);2 void operator delete(void*, void*);

palcement new 的主要用途就是反复使用一块较大的动态分配的内存来构造不同类型的对象或者他们的数组。placement new构造起来的对象或其数组,要显示的调用他们的析构函数来销毁,千万不要使用delete。

二、示例

plain new

char *getMemory(unsigned long size)   {        char * p = new char[size];       return p; }   void main(void)   {    try{           char * p = getMemory(1000000);    // 可能发生异常        // ...           delete [] p;       }       catch(const std::bad_alloc & ex)       {        cout << ex.what();    }   }

nowthrow new

void func(unsinged long length)   {    unsinged char * p = new(nothrow) unsinged char[length];       // 在使用这种new时要加(nothrow) ,明示不使用异常处理 。     if (p == NULL)  // 因不抛异常,故定要检查        cout << "allocte failed !";           // ...       delete [] p;}

placement new

void main()   {      using namespace std;       char * p = new(nothrow) char [4];       if (p == NULL)       {        cout << "allocte failed" << endl;          exit( -1 );    }       // ...       long * q = new (p) long(1000);       delete []p;    // 只释放 p,不要用q释放。}

p和q仅仅是首址相同,所构建的对象可以类型不同。所“放置”的空间应小于原空间,以防不测。当”放置new”超过了申请的范围,Debug版下会挂机,但Release版竟然能运行而不出错!

该运算符的作用是:只要第一次分配成功,不再担心分配失败。

void main()   {    using namespace std;       char * p = new(nothrow) char [100];       if (p == NULL)       {          cout << "allocte failed" << endl;        exit(-1);    }       long * q1 = new (p) long(100);       // 使用q1  ...       int * q2 = new (p) int[100/sizeof(int)];       // 使用q2 ...       ADT * q3 = new (p) ADT[100/sizeof(ADT)];       // 使用q3  然后释放对象 ...       delete [] p;    // 只释放空间,不再析构对象。}

注意:使用该运算符构造的对象或数组,一定要显式调用析构函数,不可用delete代替析构,因为placement new 的对象的大小不再与原空间相同。

void main()   {      using namespace std;       char * p = new(nothrow) char [sizeof(ADT)+2];       if (p == NULL)       {          cout << "allocte failed" << endl;        exit(-1);     }     // ...     ADT * q = new (p) ADT;     // ...     // delete q; // 错误    q->ADT::~ADT();  // 显式调用析构函数,仅释放对象    delete [] p;     // 最后,再用原指针来释放内存}

placement new 的主要用途就是可以反复使用一块已申请成功的内存空间。这样可以避免申请失败的徒劳,又可以避免使用后的释放。

特别要注意的是对于 placement new 绝不可以调用的delete, 因为该new只是使用别人替它申请的地方(只是个租房户,不是房主。无权将房子卖掉)。释放内存是nothrow new的事,即要使用原来的指针释放内存。

图片描述

转载地址:http://unyqa.baihongyu.com/

你可能感兴趣的文章
数据结构(C语言版)第三章:栈和队列
查看>>
Stopping and/or Restarting an embedded Jetty in...
查看>>
Oracle存储过程中的数据集输入参数
查看>>
vsftp 配置
查看>>
VCSA中配置时间和时区,实测至6.5适用
查看>>
高并发IM系统架构优化实践
查看>>
产品经理教你玩转阿里云负载均衡SLB系列(一):快速入门--什么是负载均衡
查看>>
有关linux--进程组、会话、守护进程详解
查看>>
我的友情链接
查看>>
monkeyrunner运行Python脚本来检查apk渠道和验证是否可以调用微信
查看>>
github获得SSH Key解决Permission denied (publickey)问题
查看>>
用java代码编写Oracle存储过程
查看>>
APACHE转发
查看>>
android-market-api
查看>>
解決 yum update錯誤:[Errno -1] Metadata file does not match checksum
查看>>
我的友情链接
查看>>
『Data Science』R语言学习笔记,获取数据
查看>>
rails中n秒页面自动跳转
查看>>
我的友情链接
查看>>
忘记root用户密码怎么办?
查看>>