Контейнер повышения указателя - не может сохранить класс в качестве ключа для ptr_map - PullRequest
0 голосов
/ 05 января 2012

Я пытался поиграть с библиотекой Boost Pointer Container и использовать их учебные примеры, чтобы почувствовать библиотеку.Может быть, я что-то упустил, но я не могу сохранить простой класс, который я определил как ключ к ptr_map.Тем не менее, тот же ключ работает для ptr_set.

<!-- language: lang-cpp -->

#include <boost/ptr_container/ptr_map.hpp>
#include <boost/ptr_container/ptr_set.hpp>
class mammal
{
  public:
    mammal(string name) : _name(name) {}
    mammal(string name, int age) : _name(name), _age(age) {}

    int age() const
    {
        return _age;
    }

    string name() const
    {
        return _name;
    }

    void eat() { cout << "Not Hungry..." << endl; }

    bool operator<(const mammal& l) const
    {
        cout << " Using Mammal Impl" << endl;
        if (name() == l.name())
            return age() < l.age();

        return name() < l.name();
    }

  private:
    string _name;
    int _age;
};


int main()
{
   std::string bobo = "bobo",
               anna = "anna",
               zublu = "zublu";

   typedef boost::ptr_set<mammal> MammalsContainer;

   MammalsContainer mammalZoo;

   mammal* m1 = new mammal(zublu, 14);
   mammal* m2 = new mammal(bobo, 31);
   mammal* m3 = new mammal(anna, 441);
   mammal* m4 = new mammal(bobo, 21);

   mammalZoo.insert(m1);
   mammalZoo.insert(m2);
   mammalZoo.insert(m3);
   mammalZoo.insert(m4);

   for (MammalsContainer::iterator i = mammalZoo.begin(); i != mammalZoo.end(); ++i)
   {
      cout << " Mammal Name: " << i->name() << " Age: " << i->age() << endl;
   }

   return 0;
}

Это приводит к следующему правильному выводу:

Mammal Name: anna Age: 441
Mammal Name: bobo Age: 21
Mammal Name: bobo Age: 31
Mammal Name: zublu Age: 14

Однако, если я переключаю MammalsContainer на ptr_map, он не '• Даже компиляция:

    typedef boost::ptr_map<mammal, int> MammalsContainer;

    mammalZoo.insert(m1, 1);
    mammalZoo.insert(m2, 2);
    mammalZoo.insert(m3, 3);
    mammalZoo.insert(m4, 4);

Приводит к следующим ошибкам компиляции:

ptrcontainer.cpp:125: error: no matching function for call to ‘boost::ptr_map<mammal, 
int, std::less<mammal>, boost::heap_clone_allocator, std::allocator<std::pair<const 
mammal, void*> > >::insert(mammal*&, int)’

/usr/local/include/boost/ptr_container/ptr_map_adapter.hpp:548: note: candidates are: 
std::pair<typename boost::ptr_container_detail::ptr_map_adapter_base<T, VoidPtrMap, 
CloneAllocator, Ordered>::iterator, bool> boost::ptr_map_adapter<T, VoidPtrMap, 
CloneAllocator, Ordered>::insert(typename 
boost::ptr_container_detail::ptr_map_adapter_base<T, VoidPtrMap, CloneAllocator, 
Ordered>::key_type&, typename boost::ptr_container_detail::ptr_map_adapter_base<T, 
VoidPtrMap, CloneAllocator, Ordered>::mapped_type) [with T = int, VoidPtrMap = 
std::map<mammal, void*, std::less<mammal>, std::allocator<std::pair<const mammal, 
void*> > >, CloneAllocator = boost::heap_clone_allocator, bool Ordered = true]

Я прочитал всю документацию / примеры на странице повышения Pointer Container вместе со множеством StackOverflowпроблемы, связанные с ptr_map;кажется, что каждый случай использования предполагает использование простого примитивного ключа, такого как std :: string или int.Это работает, если у меня есть строка / int Key в вышеприведенном примере и я храню класс Mammal в качестве значения.

То, что я действительно хочу построить, это иерархия объектов Key и возможность предоставить специализированный оператор

 std::map<boost::shared_ptr<KeyObj>, ValueObj> 

, но меня привели к тому, что библиотека ptr_container стала более эффективным решением, которое обрабатывает полиморфизм в ассоциативных контейнерах.Или, может быть, я пришел к такому выводу.

1 Ответ

2 голосов
/ 05 января 2012

В ptr_map только значение (second) сохраняется указателем. Ключ по-прежнему сохраняется по значению, и вы передаете указатель, а не значение.

...