关于placement new
来源:互联网 发布:js将字符串转化为日期 编辑:程序博客网 时间:2024/06/05 17:02
placement new 是重载operator new的一个标准、全局的版本,它不能被自定义的版
本代替(不像普通的operator new和operator delete能够被替换成用户自定义的版
本)。
它的原型如下:
void *operator new( size_t, void *p ) throw() { return p; }
首先我们区分下几个容易混淆的关键词:new、operator new、placement new
new和delete操作符我们应该都用过,它们是对堆中的内存进行申请和释放,而这两
个都是不能被重载的。要实现不同的内存分配行为,需要重载operator new,而不是
new和delete。
看如下代码:
class MyClass {…};
MyClass * p=new MyClass;
这里的new实际上是执行如下3个过程:
1. 调用operator new分配内存 ;2. 调用构造函数生成类对象;3. 返回相应指针。
operator new就像operator+一样,是可以重载的。如果类中没有重载operator new
,那么调用的就是全局的::operator new来完成堆的分配。同理,operator new[]、
operator delete、operator delete[]也是可以重载的,一般你重载的其中一个,那
么最后把其余的三个都重载一遍。
至于placement new才是本文的重点。其实它也只是operator new的一个重载的版本
,只是我们很少用到它。如果你想在已经分配的内存中创建一个对象,使用new时行
不通的。也就是说placement new允许你在一个已经分配好的内存中(栈或者堆中)
构造一个新的对象。原型中void*p实际上就是指向一个已经分配好的内存缓冲区的的
首地址。
我们知道使用new操作符分配内存需要在堆中查找足够大的剩余空间,这个操作速度
是很慢的,而且有可能出现无法分配内存的异常(空间不够)。 placement new就可
以解决这个问题。我们构造对象都是在一个预先准备好了的内存缓冲区中进行,不需
要查找内存,内存分配的时间是常数;而且不会出现在程序运行中途出现内存不足的
异常。所以,placement new非常适合那些对时间要求比较高,长时间运行不希望被
打断的应用程序。
使用方法如下:
1. 缓冲区提前分配
可以使用堆的空间,也可以使用栈的空间,所以分配方式有如下两种:
class MyClass {…};
char *buf=new char[N*sizeof(MyClass)+sizeof(int)];或者char buf[N*sizeof(MyClass
)+sizeof(int)];
2. 对象的构造
MyClass * pClass=new(buf) MyClass;
3. 对象的销毁
一旦这个对象使用完毕,你必须显式的调用类的析构函数进行销毁对象。但此时内存
空间不会被释放,以便其他的对象的构造。
pClass->~MyClass();
4. 内存的释放
如果缓冲区在堆中,那么调用delete[] buf;进行内存的释放;如果在栈中,那么在
其作用域内有效,跳出作用域,内存自动释放。
注意:
在C++标准中,对于placement operator new []有如下的说明: placement operator
new[] needs implementation-defined amount of additional storage to save
a size of array. 所以我们必须申请比原始对象大小多出sizeof(int)个字节来存放
对象的个数,或者说数组的大小。
使用方法第二步中的new才是placement new,其实是没有申请内存的,只是调用了构
造函数,返回一个指向已经分配好的内存的一个指针,所以对象销毁的时候不需要调
用delete释放空间,但必须调用析构函数销毁对象。
btw 另一篇
placement new是一个很神奇的东西,可以在你指定的位置存放对象、申请内存等。
什么是placement new?
形式上,这就是一个带参数的new:new(T2 x) T,或者new(T2 x) T[]。编译器会调
用operator new(size_t, T2 )来分配内存。编译器会提供一个operator new(size_t
, void*)的实现,返回你传入参数指定的地址,需要引用头文件<new>。例:
#include <new>
char *ptr = ...;
long l=...;
A *p = new (ptr) A();
B *q = new (l) B();
需要注意的是,你指定的地址必须有足够大的空间,并且地址要按4/8字节对齐。什
么?你没听说过字节对齐?那最好不要用placement new了。。。
重载自己的operator new
我们可以重载一些自己的operator new,还是挺好玩的:
#include <new>
char ptr[MAX];
operator new(size_t, int n) {
return ptr+n*sizeof(A);
}
A *p = new(100) A();
nothrow new
实际上,nothrow new也是用上述办法实现的。该方式的new在分配内存失败的时候返
回NULL指针,而不是抛std::bad_alloc异常。
T *p = new(std::nothrow) T;
也可以自己定义一个nothrow_t对象来做上面的事情:
std::nothrow_t nt;
T *p = new(nt) T;
需要显示调用析构函数
placement new必须显式调用析构函数来释放资源。
void someCode()
{
char memory[sizeof(Fred)];
void* p = memory;
Fred* f = new(p) Fred();
// ...
f->~Fred();// 显式调用定位放置的对象的析构函数
}
如果你分配的空间来自一个数组,那就不用管了,如果是来自动态分配的空间,还要
记得释放这些空间。
- 关于placement new
- 关于placement new
- 关于placement new
- 关于placement new操作
- 关于new operator, operator new, placement new
- 关于new delete和placement new,placement delete
- 关于C++的placement new和placement delete
- 关于 Placement new 的用法,求解!
- 关于C++中的placement new操作符
- placement new
- placement new
- placement new
- placement new
- placement new
- placement new
- placement new
- placement new
- placement new
- Android基础篇-自动获取屏幕的尺寸及密度
- 开始研究KUI
- 逛校友录
- android下apk反编译问题总结
- 转载 某老板的文
- 关于placement new
- CAS单点登录(SSO)完整教程
- 程序是写出来的
- ARP欺骗
- c#非递归遍历所有节点
- 关于Hadoop数据块Miss在页面提示的问题
- 扩展Tomcat使用memcached存放session信息
- Microsoft 数据访问技术总结
- Linux 中 crontab 详解及示例