我正在尝试实现一个自定义内存管理器,我想知道是否有更好的方法来实现这个函数,因为当我被问及无效指针算术时,有些人认为如果我在C ++中有一个void *,事情是非常错误的。
// allocates a page of memory.
void ObjectAllocator::allocatePage()
{
//if(OAStats_.PagesInUse_ >= Config_.MaxPages_)
//throw exception
void* buffer = ::operator new(OAStats_.PageSize_); // allocate memory, no constructor call.
// =============== Setup the PageList_ ===============
GenericObject* pNewNode = ::new(buffer) GenericObject(); // Construct GenericObject for the pagelist.
pNewNode->Next = PageList_->Next; // pNewNode points to wherever PageList_ pointed to.
PageList_->Next = pNewNode; // PageList_ points to pNewNode
pNewNode = NULL; // dont need this handle anymore
buffer = static_cast<char*>(buffer) + sizeof(GenericObject); // move pointer to point after the generic object.
// =============== Setup the FreeList_ ===============
for(int i=0;i<Config_.ObjectsPerPage_;++i)
{
static GenericObject* pPreviousNode = NULL; // static variable to hold the previous node
pNewNode = ::new(buffer) GenericObject(); // Construct GenericObject for the freelist.
pNewNode->Next = pPreviousNode;
pPreviousNode = pNewNode;
buffer = static_cast<char*>(buffer) + OAStats_.ObjectSize_; // move pointer by ObjectSize.
++OAStats_.FreeObjects_;
}
FreeList_->Next = pNewNode;
++OAStats_.PagesInUse_;
++OAStats_.Allocations_;
}
答案 0 :(得分:10)
如果你需要一块内存用于存储字符串(8位ANSI),那么将指向该缓冲区的指针声明为char并对其进行操作是有意义的。
在你的情况下,你需要一块“blob”的内存块,它没有固有的类型,所以你正确地选择了void *来代表那个blob。
现在您需要按某个对象的大小增加该指针。由于显而易见的原因,您无法对void指针执行算术运算,那么您该怎么办?投吧。没有羞耻。
答案 1 :(得分:6)
在C ++中,在原始字节上,使用char *,并且不要想你自己。这是正确的事(tm)。特别是如果你把它包装在一个更高级别的结构中,就像你一样。
答案 2 :(得分:2)
虚空*没有任何内在错误。然而,我们经常看到的是来自C的人,当他们应该做其他事情时,过度使用void *。如果你正在管理原始内存blob,那么void *是完全合适的。但是,很少有任何其他理由去做。