c++中new的三種用法詳細(xì)解析
一. 簡(jiǎn)介
new有三種使用方式:plain new,nothrow new和placement new。
(1)plain new顧名思義就是普通的new,就是我們慣常使用的new。在C++中是這樣定義的:
void* operator new(std::size_t) throw(std::bad_alloc);
void operator delete(void *) throw();
提示:plain new在分配失敗的情況下,拋出異常std::bad_alloc而不是返回NULL,因此通過(guò)判斷返回值是否為NULL是徒勞的。
(2)nothrow new是不拋出異常的運(yùn)算符new的形式。nothrow new在失敗時(shí),返回NULL。定義如下:
void * operator new(std::size_t,const std::nothrow_t&) throw();
void operator delete(void*) throw();
(3)placement new意即“放置”,這種new允許在一塊已經(jīng)分配成功的內(nèi)存上重新構(gòu)造對(duì)象或?qū)ο髷?shù)組。placement new不用擔(dān)心內(nèi)存分配失敗,因?yàn)樗静环峙鋬?nèi)存,它做的唯一一件事情就是調(diào)用對(duì)象的構(gòu)造函數(shù)。定義如下:
void* operator new(size_t,void*);
void operator delete(void*,void*);
提示1:palcement new的主要用途就是反復(fù)使用一塊較大的動(dòng)態(tài)分配的內(nèi)存來(lái)構(gòu)造不同類型的對(duì)象或者他們的數(shù)組。
提示2:placement new構(gòu)造起來(lái)的對(duì)象或其數(shù)組,要顯示的調(diào)用他們的析構(gòu)函數(shù)來(lái)銷毀,千萬(wàn)不要使用delete。
char* p = new(nothrow) char[100];
long *q1 = new(p) long(100);
int *q2 = new(p) int[100/sizeof(int)];
二.實(shí)例
1.plain new/delete.普通的new
定義如下:
void *operator new(std::size_t) throw(std::bad_alloc);
void operator delete(void*) throw();
注:標(biāo)準(zhǔn)C++ plain new失敗后拋出標(biāo)準(zhǔn)異常std::bad_alloc而非返回NULL,因此檢查返回值是否為NULL判斷分配是否成功是徒勞的。
測(cè)試程序:
#include "stdafx.h"
#include <iostream>
using namespace std;
char *GetMemory(unsigned long size)
{
char *p=new char[size];//分配失敗,不是返回NULL
return p;
}
int main()
{
try
{
char *p=GetMemory(10e11);// 分配失敗拋出異常std::bad_alloc
//...........
if(!p)//徒勞
cout<<"failure"<<endl;
delete [] p;
}
catch(const std::bad_alloc &ex)
{
cout<<ex.what()<<endl;
}
return 0;
}
2.nothrow new/delete不拋出異常的運(yùn)算符new的形式,new失敗時(shí)返回NULL。
定義如下:
void *operator new(std::size_t,const std::nothrow_t&) throw();
void operator delete(void*) throw();
struct nothrow_t{}; const nothrow_t nothrow;//nothrow作為new的標(biāo)志性啞元
測(cè)試程序:
#include "stdafx.h"
#include <iostream>
#include <new>
using namespace std;
char *GetMemory(unsigned long size)
{
char *p=new(nothrow) char[size];//分配失敗,是返回NULL
if(NULL==p)
cout<<"alloc failure!"<<endl;
return p;
}
int main()
{
try
{
char *p=GetMemory(10e11);
//...........
if(p==NULL)
cout<<"failure"<<endl;
delete [] p;
}
catch(const std::bad_alloc &ex)
{
cout<<ex.what()<<endl;
}
return 0;
}
3.placement new/delete 主要用途是:反復(fù)使用一塊較大的動(dòng)態(tài)分配成功的內(nèi)存來(lái)構(gòu)造不同類型的對(duì)象或者它們的數(shù)組。例如可以先申請(qǐng)一個(gè)足夠大的字符數(shù)組,然后當(dāng)需要時(shí)在它上面構(gòu)造不同類型的對(duì)象或數(shù)組。placement new不用擔(dān)心內(nèi)存分配失敗,因?yàn)樗静环峙鋬?nèi)存,它只是調(diào)用對(duì)象的構(gòu)造函數(shù)。
測(cè)試程序:
#include "stdafx.h"
#include <iostream>
#include <new>
using namespace std;
class ADT
{
int i;
int j;
public:
ADT()
{
}
~ADT()
{
}
};
int main()
{
char *p=new(nothrow) char[sizeof(ADT)+2];
if(p==NULL)
cout<<"failure"<<endl;
ADT *q=new(p) ADT; //placement new:不必?fù)?dān)心失敗
// delete q;//錯(cuò)誤!不能在此處調(diào)用delete q;
q->ADT::~ADT();//顯示調(diào)用析構(gòu)函數(shù)
delete []p;
return 0;
}
注:使用placement new構(gòu)造起來(lái)的對(duì)象或數(shù)組,要顯式調(diào)用它們的析構(gòu)函數(shù)來(lái)銷毀(析構(gòu)函數(shù)并不釋放對(duì)象的內(nèi)存),千萬(wàn)不要使用delete.這是因?yàn)閜lacement new構(gòu)造起來(lái)的對(duì)象或數(shù)組大小并不一定等于原來(lái)分配的內(nèi)存大小,使用delete會(huì)造成內(nèi)存泄漏或者之后釋放內(nèi)存時(shí)出現(xiàn)運(yùn)行時(shí)錯(cuò)誤。
相關(guān)文章
C++實(shí)現(xiàn)重載矩陣的部分運(yùn)算符
這篇文章主要為大家詳細(xì)介紹了如何利用C++實(shí)現(xiàn)重載矩陣的部分運(yùn)算符,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)C++有一定幫助,需要的可以參考一下2022-10-10
c++ 單線程實(shí)現(xiàn)同時(shí)監(jiān)聽多個(gè)端口
這篇文章主要介紹了c++ 單線程實(shí)現(xiàn)同時(shí)監(jiān)聽多個(gè)端口的方法,幫助大家更好的理解和學(xué)習(xí)使用c++,感興趣的朋友可以了解下2021-03-03
基于VC中使用ForceInclude來(lái)強(qiáng)制包含stdafx.h的解決方法
本篇文章是對(duì)VC中使用ForceInclude來(lái)強(qiáng)制包含stdafx.h的解決方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
C++?requires關(guān)鍵字簡(jiǎn)單介紹
requires?是?C++20?中引入的一個(gè)新關(guān)鍵字,用于在函數(shù)模板或類模板中聲明所需的一組語(yǔ)義要求,它可以用來(lái)限制模板參數(shù),類似于?typename?和?class?關(guān)鍵字,這篇文章主要介紹了C++?requires關(guān)鍵字簡(jiǎn)介,需要的朋友可以參考下2023-05-05
C++ GDI實(shí)現(xiàn)圖片格式轉(zhuǎn)換
GDI+(Graphics Device Interface Plus)是一種用于圖形繪制和圖像處理的應(yīng)用程序編程接口(API),在Windows平臺(tái)上廣泛使用,本文就來(lái)介紹一下如何使用GDI實(shí)現(xiàn)圖片格式轉(zhuǎn)換吧2023-12-12
C++實(shí)現(xiàn)判斷字符串是否回文實(shí)例解析
這篇文章主要介紹了C++實(shí)現(xiàn)判斷字符串是否回文,其中采用了數(shù)據(jù)結(jié)構(gòu)中棧以及過(guò)濾字符等技術(shù),,需要的朋友可以參考下2014-07-07
C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)圖的創(chuàng)建與遍歷實(shí)驗(yàn)示例
這篇文章主要為大家介紹了C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)圖的創(chuàng)建與遍歷實(shí)驗(yàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06
C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單飛機(jī)大戰(zhàn)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單飛機(jī)大戰(zhàn),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-02-02

