Using Both Map and List for Same Objects

后端 未结 2 1535
抹茶落季
抹茶落季 2021-01-23 13:53

I\'m trying to use both a list and an unordered_map to store the same set of objects. I\'m new to C++, so still getting comfortable with iterators.

Say I have the follow

相关标签:
2条回答
  • 2021-01-23 14:04

    While Barry's approach is good, there is another one, more advanced and complicated. You can put your data object, (integer) key, and all bookkeeping bits in a single chunk of memory. Thus data locality will be improved and pressure on memory allocator will be less. Example, using boost::intrusive:

    #include <boost/intrusive/list.hpp>
    #include <boost/intrusive/unordered_set.hpp>
    #include <array>
    
    using namespace boost::intrusive;
    
    class Foo {
        // bookkeeping bits
        list_member_hook<> list_hook;
        unordered_set_member_hook<> set_hook;
    
        const int key;
        // some payload...
    
    public:
        // there is even more options to configure container types
        using list_type = list<Foo, member_hook<Foo, list_member_hook<>, &Foo::list_hook>>;
        using set_type = unordered_set<Foo, member_hook<Foo, unordered_set_member_hook<>, &Foo::set_hook>>;
    
        Foo(int key): key(key) {};
        bool operator ==(const Foo &rhs) const {
            return key == rhs.key;
        }
        friend std::size_t hash_value(const Foo &foo) {
            return std::hash<int>()(foo.key);
        }
    };
    
    class Bar {
        Foo::list_type list;
    
        std::array<Foo::set_type::bucket_type, 17> buckets;
        Foo::set_type set{Foo::set_type::bucket_traits(buckets.data(), buckets.size())};
    
    public:
        template<typename... Args>
        Foo &emplace(Args&&... args) {
            auto foo = new Foo(std::forward<Args>(args)...);
            // no more allocations
            list.push_front(*foo);
            set.insert(*foo);
            return *foo;
        }
        void pop(const Foo &foo) {
            set.erase(foo);
            list.erase(list.iterator_to(foo));
            // Lifetime management fun...
            delete &foo;
        }
    };
    
    int main() {
        Bar bar;
        auto &foo = bar.emplace(42);
        bar.pop(foo);
    }
    

    Measure how good are both algorithms on your data. My idea may give you nothing but greater code complexity.

    0 讨论(0)
  • 2021-01-23 14:24

    If what you want to do is this:

    Is it possible to look up an object by key, and then generate a list iterator from the reference of the object (or from the unordered_map generator?)

    Then you can take advantage of the fact that list iterators aren't invalidated on insertion or erase (unless you erase that particular iterator) and reorganize your structures like this:

    std::list<Test> list;
    std::unordered_map<int, std::list<Test>::iterator> map;
    
    map.insert(std::make_pair(101, 
        list.insert(list.end(), t1)));
    map.insert(std::make_pair(102, 
        list.insert(list.end(), t2)));
    map.insert(std::make_pair(103, 
        list.insert(list.end(), t3)));
    

    That way your map lookup gives you exactly what you want: a list iterator.

    0 讨论(0)
提交回复
热议问题