(轉(zhuǎn))遵循placement new的用法規(guī)范
標(biāo)準(zhǔn)的開(kāi)始
Widget * p = new Widget;
//ordinary new? //普通的new pi = new (ptr) int;
pi = new (ptr) int; //placement new
括號(hào)里的參數(shù)是一個(gè)指針,它指向一個(gè)內(nèi)存緩沖器,placement new將在這個(gè)緩沖器上分配一個(gè)對(duì)象。Placement new的返回值是這個(gè)被構(gòu)造對(duì)象的地址(比如扣號(hào)中的傳遞參數(shù))。placement new主要適用于:在對(duì)時(shí)間要求非常高的應(yīng)用程序中,因?yàn)檫@些程序分配的時(shí)間是確定的;長(zhǎng)時(shí)間運(yùn)行而不被打斷的程序;以及執(zhí)行一個(gè)垃圾收集器(garbage collector)。
使用方法
在很多情況下,placement new的使用方法和其他普通的new有所不同。這里提供了它的使用步驟。
第一步? 緩存提前分配
為了保證通過(guò)placement new使用的緩存區(qū)的memory alignmen(內(nèi)存隊(duì)列)正確準(zhǔn)備,使用普通的new來(lái)分配它:
class Task ;
char * buff = new [sizeof(Task)]; //分配內(nèi)存
(請(qǐng)注意auto或者static內(nèi)存并非都正確地為每一個(gè)對(duì)象類(lèi)型排列,所以,你將不能以placement new使用它們。)
第二步:對(duì)象的分配
在剛才已分配的緩存區(qū)調(diào)用placement new來(lái)構(gòu)造一個(gè)對(duì)象。
Task *ptask = new(buff) Task
第三步:使用
按照普通方式使用分配的對(duì)象:
ptask->suspend();
ptask->resume();
//...
第四步:對(duì)象的毀滅
一旦你使用完這個(gè)對(duì)象,你必須調(diào)用它的析構(gòu)函數(shù)來(lái)毀滅它。按照下面的方式調(diào)用析構(gòu)函數(shù):
ptask->~Task(); //調(diào)用外在的析構(gòu)函數(shù)
第五步:釋放
你可以反復(fù)利用緩存并給它分配一個(gè)新的對(duì)象(重復(fù)步驟2,3,4)如果你不打算再次使用這個(gè)緩存,你可以象這樣釋放它:
delete [] buff;
跳過(guò)任何步驟就可能導(dǎo)致運(yùn)行時(shí)間的崩潰,內(nèi)存泄露,以及其它的意想不到的情況。如果你確實(shí)需要使用placement new,請(qǐng)認(rèn)真遵循以上的步驟。
本文作者:Danny Kalev是具有14年經(jīng)驗(yàn)的系統(tǒng)分析員和軟件工程師,他主要專(zhuān)長(zhǎng)于C++和面象對(duì)象程序設(shè)計(jì)。