7

我正在尝试在共享内存中创建一个 unordered_map。我正在使用分配器来达到目的。

编码

void *addr;
void *pool;
int shmid;

template<class T>
class MyPoolAlloc {
private:
public:
    typedef size_t     size_type;
    typedef ptrdiff_t  difference_type;
    typedef T*         pointer;
    typedef const T*   const_pointer;
    typedef T&         reference;
    typedef const T&   const_reference;
    typedef T          value_type;

   template<class X>
   struct rebind
   { typedef MyPoolAlloc<X> other; };

   MyPoolAlloc() throw() {
   }
   MyPoolAlloc(const MyPoolAlloc&) throw()  {
   }

   template<class X>
   MyPoolAlloc(const MyPoolAlloc<X>&) throw() {
   }

   ~MyPoolAlloc() throw() {
   }

  pointer address(reference __x) const { return &__x; }

  const_pointer address(const_reference __x) const { return &__x; }

  pointer allocate(size_type __n, const void * hint = 0) {
      pointer tmp = static_cast<T*>(addr);
      addr = (void*)((char*)addr + __n);
      return tmp;
  }

  void deallocate(pointer __p, size_type __n) {
      // pMyPool->Free(reinterpret_cast<void *>(__p));
  }

 size_type max_size() const throw() {
    //return size_t(-1) / sizeof(T);
  }

 void construct(pointer __p, const T& __val) {
     ::new(__p) T(__val);
  }

 void destroy(pointer __p) {
    //__p->~T();
  }
};

typedef std::unordered_map<int, int, std::hash<int>, std::equal_to<int>,  MyPoolAlloc<std::pair<const int,int>> > Map;

int main ()
{
    shmid = shmget(shm_key, 1000, IPC_CREAT | IPC_EXCL | 644);
    if(shmid == -1){
            std::cerr << "Failed to create the shared segment." << strerror(errno)<< std::endl;
            exit(-1);
    }

    addr = shmat(shmid, NULL, 0);
    pool = addr;
    if(addr == (void*)-1){
            std::cerr << "Failed to attach the segment to the process." << std::endl;
            shmctl(shmid, IPC_RMID, 0);
            exit(-1);
    }

    Map *m = new(pool) Map;

    m->insert(std::pair<int, int>(2,4));

    shmdt(addr);
    shmctl(shmid, IPC_RMID, 0);

    return 0;
}

我正在为共享内存地址上的映射分配内存。我想对于元素,将使用分配器类的 allocate() 函数。但我无法弄清楚如何使用 allocate 函数为地图元素分配内存。我在尝试插入地图的行处遇到算术异常。

有人可以帮助了解内存布局应该如何吗?

谢谢。

4

1 回答 1

7

正如我上次你问到这个问题时所说的,你不能将指针存储在共享内存中并期望它们在多个进程中可用。和以前一样,我建议boost::unordered_mapboost::interprocess::allocator.

如果您真的想编写自己的分配器,则需要某种自定义指针,将偏移量存储到共享内存块中,并能够从中重建正确的地址。我不建议您自己尝试这样做:除了非常繁琐之外,您可能会发现某些实现std::unordered_map不能与非常规指针类型一起正常工作。事实上,我怀疑为共享内存创建一个符合标准的分配器可能是不可能的。否则,Boost 分配器肯定可以被所有标准容器使用,而不仅仅是特殊的 Boost 版本。

顺便说一句,您不应该__p在自己的代码中使用保留名称,例如(包含双下划线);它们只能由标准库实现使用。

于 2012-08-07T08:59:17.327 回答