什么是placement new?
所謂placement new就是在用戶指定的內(nèi)存位置上構(gòu)建新的對象,這個構(gòu)建過程不需要額外分配內(nèi)存,只需要調(diào)用對象的構(gòu)造函數(shù)即可。
舉例來說:
class foo{};
foo* pfoo = new foo;
pfoo指向的對象的地址你是不能決定的,因為new已經(jīng)為你做了這些工作。第一步分配內(nèi)存,第二步調(diào)用類的構(gòu)造函數(shù)。
而placement new是怎么做的呢,說白了就是把原本new做的兩步工作分開來。第一步你自己分配內(nèi)存,第二步你調(diào)用類的構(gòu)造函數(shù)在自己分配的內(nèi)存上構(gòu)建新的對象。
placement new的好處:
1)在已分配好的內(nèi)存上進(jìn)行對象的構(gòu)建,構(gòu)建速度快。
2)已分配好的內(nèi)存可以反復(fù)利用,有效的避免內(nèi)存碎片問題。
關(guān)于Placement new的標(biāo)準(zhǔn)用法,網(wǎng)上有一篇文章講得很清楚,我這里再轉(zhuǎn)述一下。
class?Foo
{
????char?cc;
????float?f;
public:
????void?print()?{?std::cout?<<?"ADDR:?"?<<?this?<<?std::endl;?}
????void?set_f(?float?_f?)?{?std::cout?<<?"set?f?val?:?"?<<?_f?<<?std::endl;?f?=?_f;?}
????void?get_f()?{?std::cout?<<?"get?f?val?:?"?<<?f?<<?std::endl;?}
};
1)分配內(nèi)存
char*?buff?=?new?char[?sizeof(Foo)?*?N?];
memset(?buff,?0,?sizeof(Foo)*N?);
2)構(gòu)建對象
Foo*?pfoo?=?new?(buff)Foo;
3)使用對象
pfoo->print();
pfoo->set_f(1.0f);
pfoo->get_f();
4)析構(gòu)對象,顯式的調(diào)用類的析構(gòu)函數(shù)。
pfoo->~Foo();
5)銷毀內(nèi)存
delete?[]?buff;
上面5個步驟是標(biāo)準(zhǔn)的placement new的使用方法。
對于buff這塊內(nèi)存可以反復(fù)使用,只要重復(fù)2)、3)、4)步驟即可。
placement new還可以解決的一個問題是建立帶參數(shù)的構(gòu)造函數(shù)對象數(shù)組。
代碼示例如下:
?1?class?CPong
?2?{
?3?public:
?4?????CPong(?int?m?)?:?v(m)?{?std::cout?<<?"CPong?ctor."?<<?std::endl;?}
?5?
?6?private:
?7?????int?v;
?8?};
?9?char*?pong?=?new?char[?sizeof(CPong)?*?10?];
10?CPong*?pp?=?(CPong*)pong;
11?
12?for?(?int?i=0;?i<10;?++i?)
13?{
14?????new?(pp+i)CPong(i);
15?}
16?
17?for?(?int?j=0;?j<10;?++j?)
18?{
19???? pp[j].~CPong();
20?}
21?
22?delete?[]?pong;