c,c++中如何做到在指定内存地址创建对象实例

ithiker 2013-06-11 04:21:52
如题,最好有小例子说明
...全文
1389 17 打赏 收藏 转发到动态 举报
写回复
用AI写文章
17 条回复
切换为时间正序
请发表友善的回复…
发表回复
wbbforeverlove 2014-02-13
  • 打赏
  • 举报
回复
可以啊,上面不是说了吗 用 void* buf = reinterpret_cast<void*> (0xF00F);
大尾巴猫 2013-06-12
  • 打赏
  • 举报
回复
引用 9 楼 gigglesun 的回复:
多谢大家的回复,大家回答的还是没满足我的需求。比如说我想在内存地址为p的地方创建一个int,malloc出来内存空间的开始地址并不一定和p一样,比如说malloc出来的内存开始地址为q,int *f=new(q)int创建出来一个int,但是开始地址为q。不知道我这样要求算不算合理。
你这个要求好像做不到。 而且你得到的这个地址是虚拟的地址,并不是真正物理内存地址,是操作系统经过映射的虚拟内存地址。
ithiker 2013-06-12
  • 打赏
  • 举报
回复
多谢大家的回复,大家回答的还是没满足我的需求。比如说我想在内存地址为p的地方创建一个int,malloc出来内存空间的开始地址并不一定和p一样,比如说malloc出来的内存开始地址为q,int *f=new(q)int创建出来一个int,但是开始地址为q。不知道我这样要求算不算合理。
rocktyt 2013-06-12
  • 打赏
  • 举报
回复
引用 14 楼 ananluowei 的回复:
[quote=引用 11 楼 rocktyt2 的回复:] [quote=引用 9 楼 gigglesun 的回复:] 多谢大家的回复,大家回答的还是没满足我的需求。比如说我想在内存地址为p的地方创建一个int,malloc出来内存空间的开始地址并不一定和p一样,比如说malloc出来的内存开始地址为q,int *f=new(q)int创建出来一个int,但是开始地址为q。不知道我这样要求算不算合理。
没看懂 能说明白点吗?是想改变地址偏移呢还是想用内存池?[/quote] 没看明白楼主的意思吗? 他需要指定一个地址,比如0x00100000,在这个地址分配1个int。 当然是做不到的。[/quote]终于明白了……这个果然还是做不到吧……
大尾巴猫 2013-06-12
  • 打赏
  • 举报
回复
引用 11 楼 rocktyt2 的回复:
[quote=引用 9 楼 gigglesun 的回复:] 多谢大家的回复,大家回答的还是没满足我的需求。比如说我想在内存地址为p的地方创建一个int,malloc出来内存空间的开始地址并不一定和p一样,比如说malloc出来的内存开始地址为q,int *f=new(q)int创建出来一个int,但是开始地址为q。不知道我这样要求算不算合理。
没看懂 能说明白点吗?是想改变地址偏移呢还是想用内存池?[/quote] 没看明白楼主的意思吗? 他需要指定一个地址,比如0x00100000,在这个地址分配1个int。 当然是做不到的。
我看你有戏 2013-06-12
  • 打赏
  • 举报
回复

#include <iostream>

using namespace std;

typedef char byte;


int main()

{

byte *buffer = new byte[1000];

int *pi = new(buffer)int[10];

pi[0] = 3;

cout<<(int)*buffer<<endl;

delete []buffer;

return 0;

}

//输出结果是3,说明内存的分配是在buffer里边的,其实也不能称之为内存分配,是在已经分配好的一部分内存空间里边来划分。



#include <iostream>

typedef char byte;

using namespace std;


int main()

{

byte *buffer = new byte[1000];

int *p1 = new(buffer)int[10];

p1[0] = 3;

//输出第一个元素的值

cout<<(int)*buffer<<endl;

int *p2 = new(buffer)int[10];

p2[0] = 4;

//赋值以后输出第一个元素的值

cout<<(int)*(buffer)<<endl;


//再次输出第一个元素的值

cout<<p1[0]<<endl;

//输出这些地址

cout<<(int)buffer<<endl;

cout<<p1<<endl;

cout<<p2<<endl;

delete []buffer;

return 0;

}

//发现,两次的内存分配都是从buffer的首地址开始的,所以我猜测,buffer里边仅仅允许分配一个(一组)对象的内存。



#include <iostream>

using namespace std;


class Integer

{

private:

int datum;

int a,b,c,d;

public:

Integer(int ival = 0)

{

a = b = c = d = datum = ival;

}


Integer(Integer const& rhs)

{

datum = rhs.datum;

}


Integer& operator = (Integer const& rhs)

{

datum = rhs.datum;

return *this;

}


~Integer()

{ 

cout << "~Integer" << endl; 

}


operator int () const

{

return datum;

}

};



int _tmain(int argc, _TCHAR* argv[])

{


int const ELEMENT_COUNT = 8;

unsigned char *buffer = new unsigned char [ELEMENT_COUNT * sizeof(Integer) + sizeof(int)];

Integer *data = new (buffer) Integer[ELEMENT_COUNT];

for (int i = 0; i < ELEMENT_COUNT; i++)

{

data[i].~Integer();

}

delete [] buffer;

return 0;

}
//在申请buffer的时候需要多申请sizeof(int)的内存才不会出错,说明了编译器对用户自定义类型的存储方式---后边需要有sizeof(int)字节的内存来存储分配的空间等信息。

我看你有戏 2013-06-12
  • 打赏
  • 举报
回复
Placement new 存在的理由 (1).用Placement new 解决buffer的问题 问 题描述:用new分配的数组缓冲时,由于调用了默认构造函数,因此执行效率上不佳。若没有默认构造函数则会发生编译时错误。如果你想在预分配的内存上创建 对象,用缺省的new操作符是行不通的。要解决这个问题,你可以用placement new构造。它允许你构造一个新对象到预分配的内存上。 (2).增大时空效率的问题 使用new操作符分配内存需要在堆中查找足够大的剩余空间,显然这个操作速度是很慢的,而且有可能出现无法分配内存的异常(空间不够)。 placement new 就可以解决这个问题。我们构造对象都是在一个预先准备好了的内存缓冲区中进行,不需要查找内存,内存分配的时间是常数;而且不会出现在程序运行中途出现内 存不足的异常。所以,placement new非常适合那些对时间要求比较高,长时间运行不希望被打断的应用程序。 5. 使用步骤 在很多情况下,placement new的使用方法和其他普通的new有所不同。这里提供了它的使用步骤。 第一步 缓存提前分配 有三种方式: 1.为了保证通过placement new使用的缓存区的memory alignmen(内存队列)正确准备,使用普通的new来分配它:在堆上进行分配 class Task ; char * buff = new [sizeof(Task)]; //分配内存 (请注意auto或者static内存并非都正确地为每一个对象类型排列,所以,你将不能以placement new使用它们。) 2.在栈上进行分配 class Task ; char buf[N*sizeof(Task)]; //分配内存 3.还有一种方式,就是直接通过地址来使用。(必须是有意义的地址) void* buf = reinterpret_cast<void*> (0xF00F); 第二步:对象的分配 在刚才已分配的缓存区调用placement new来构造一个对象。 Task *ptask = new (buf) Task 第三步:使用 按照普通方式使用分配的对象: ptask->memberfunction(); ptask-> member; //... 第四步:对象的析构 一旦你使用完这个对象,你必须调用它的析构函数来毁灭它。按照下面的方式调用析构函数: ptask->~Task(); //调用外在的析构函数 第五步:释放 你可以反复利用缓存并给它分配一个新的对象(重复步骤2,3,4)如果你不打算再次使用这个缓存,你可以象这样释放它: delete [] buf; 跳过任何步骤就可能导致运行时间的崩溃,内存泄露,以及其它的意想不到的情况。如果你确实需要使用placement new,请认真遵循以上的步骤。
rocktyt 2013-06-12
  • 打赏
  • 举报
回复
引用 9 楼 gigglesun 的回复:
多谢大家的回复,大家回答的还是没满足我的需求。比如说我想在内存地址为p的地方创建一个int,malloc出来内存空间的开始地址并不一定和p一样,比如说malloc出来的内存开始地址为q,int *f=new(q)int创建出来一个int,但是开始地址为q。不知道我这样要求算不算合理。
没看懂 能说明白点吗?是想改变地址偏移呢还是想用内存池?
lm_whales 2013-06-12
  • 打赏
  • 举报
回复
自定义 operator new 用类似 Placement new 的方法就可以了,例如VC的DEBUG_NEW 就提可以看MFC的DEBUG_NEW的定义
ForestDB 2013-06-11
  • 打赏
  • 举报
回复
int * p = malloc(sizeof(int)); int * q = new(p) int;
SKATE11 2013-06-11
  • 打赏
  • 举报
回复
布局NEW C++ primer plus 上说到
derekrose 2013-06-11
  • 打赏
  • 举报
回复
placement new ++
quickSort 2013-06-11
  • 打赏
  • 举报
回复
lz,你是哪里遇到这个的? primer里面的高级主题部分有这个详细的解释, 关于内存管理的。
  • 打赏
  • 举报
回复
引用 2 楼 worldy 的回复:
[quote=引用 1 楼 adlay 的回复:] 用 placement new: #include <new> A* pa = new (地址) A;
这样能行吗?你还能绕过内存管理器? 我感觉这个想法是不行,也没有任何实际意义的[/quote] stl都这么搞的
www_adintr_com 2013-06-11
  • 打赏
  • 举报
回复
这个没有绕过什么内存管理呀. placement new 的地址需要是内存已经分配好了的. 它其实相当于是在这个地址上调用一次构造函数. new A; 等价于 void* p = malloc(sizeof(A)); new (p) A; 这种 new 在自定义内存池的代码里面会大量使用.
worldy 2013-06-11
  • 打赏
  • 举报
回复
引用 1 楼 adlay 的回复:
用 placement new: #include <new> A* pa = new (地址) A;
这样能行吗?你还能绕过内存管理器? 我感觉这个想法是不行,也没有任何实际意义的
www_adintr_com 2013-06-11
  • 打赏
  • 举报
回复
用 placement new: #include <new> A* pa = new (地址) A;

65,179

社区成员

发帖
与我相关
我的任务
社区描述
C++ 语言相关问题讨论,技术干货分享,前沿动态等
c++ 技术论坛(原bbs)
社区管理员
  • C++ 语言社区
  • encoderlee
  • paschen
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
  1. 请不要发布与C++技术无关的贴子
  2. 请不要发布与技术无关的招聘、广告的帖子
  3. 请尽可能的描述清楚你的问题,如果涉及到代码请尽可能的格式化一下

试试用AI创作助手写篇文章吧