关于new、operator new和placement…

 

看起来不可能的事情在C++中总能找到解决的办法。
正如,直接调用构造函数是不可能的,然而可以用placement new 欺骗编译器调用构造函数。

1placement new 为何物?
placement new
是重载operator new 的一个标准、全局的版本,它不能够被自定义的版本代替(不像普通版本的operator newoperator delete能够被替换)。

void *operator new( size_t, void *p ) throw()
    { return p; }

placement new的执行忽略了size_t参数,只返还第二个参数。其结果是允许用户把一个对象放到一个特定的地方,达到调用构造函数的效果。
class SPort { ... }; // represents a serial port
const int comLoc = 0x00400000; // location of a port
//...
void *comAddr = reinterpret_cast<void *>(comLoc);
SPort *com1 = new (comAddr) SPort; // create object at comLoc
com1->~SPort(); //
释放

2newoperator newplacement new 一样吗?
new
:不能被重载,其行为总是一致的。它先调用operator new分配内存,然后调用构造函数初始化那段内存。
operator new
:要实现不同的内存分配行为,应该重载operator new,而不是new
delete
operator delete类似。
placement new
:只是operator new重载的一个版本。它并不分配内存,只是返回指向已经分配好的某段内存的一个指针。因此不能删除它,但需要调用对象的析构函数。

3、在已有的内存上用placement new分配数组
const int numComs = 4;
//...
SPort *comPorts = new (comAddr) SPort[numComs]; // create array
int i = numComs;
while( i )
    comPorts[--i].~SPort();

4、用Placement new 解决buffer的问题
new分配的数组缓冲时,由于调用了默认构造函数,因此执行效率上不佳。若没有默认构造函数则会发生编译时错误。用Placement new可以解决此类问题。
const size_t n = sizeof(string) * BUFSIZE;
string *sbuf = static_cast<string *>(::operator new( n ));
int size = 0;
//
此时,buffer还没有初始化,因此需要用 placement new 调用copy构造函数初始化。
void append( string buf[], int &size, const string &val )
    { new (&buf[size++]) string( val ); } // placement new
//
最后的清理
void cleanupBuf( string buf[], int size ) {
    while( size )
        buf[--size].~string(); // destroy initialized elements
    ::operator delete( buf ); // free storage
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值