#pragma once #include #include #include #include #include namespace AK { template> class HashMap { private: struct Entry { K key; V value; }; struct EntryTraits { static unsigned hash(const Entry& entry) { return KeyTraits::hash(entry.key); } static bool equals(const Entry& a, const Entry& b) { return KeyTraits::equals(a.key, b.key); } static void dump(const Entry& entry) { kprintf("key="); KeyTraits::dump(entry.key); kprintf(" value="); Traits::dump(entry.value); } }; public: HashMap() {} bool is_empty() const { return m_table.is_empty(); } int size() const { return m_table.size(); } int capacity() const { return m_table.capacity(); } void clear() { m_table.clear(); } void set(const K& key, const V& value) { m_table.set({ key, value }); } void set(const K& key, V&& value) { m_table.set({ key, move(value) }); } void remove(const K& key) { auto it = find(key); if (it != end()) m_table.remove(it); } void remove_one_randomly() { m_table.remove(m_table.begin()); } typedef HashTable HashTableType; typedef typename HashTableType::Iterator IteratorType; typedef typename HashTableType::ConstIterator ConstIteratorType; IteratorType begin() { return m_table.begin(); } IteratorType end() { return m_table.end(); } IteratorType find(const K& key) { return m_table.find(KeyTraits::hash(key), [&](auto& entry) { return KeyTraits::equals(key, entry.key); }); } template IteratorType find(unsigned hash, Finder finder) { return m_table.find(hash, finder); } ConstIteratorType begin() const { return m_table.begin(); } ConstIteratorType end() const { return m_table.end(); } ConstIteratorType find(const K& key) const { return m_table.find(KeyTraits::hash(key), [&](auto& entry) { return KeyTraits::equals(key, entry.key); }); } template ConstIteratorType find(unsigned hash, Finder finder) const { return m_table.find(hash, finder); } void ensure_capacity(int capacity) { m_table.ensure_capacity(capacity); } void dump() const { m_table.dump(); } Optional::PeekType> get(const K& key) const { auto it = find(key); if (it == end()) return {}; return (*it).value; } bool contains(const K& key) const { return find(key) != end(); } void remove(IteratorType it) { m_table.remove(it); } V& ensure(const K& key) { auto it = find(key); if (it == end()) set(key, V()); return find(key)->value; } Vector keys() const { Vector list; list.ensure_capacity(size()); for (auto& it : *this) list.unchecked_append(it.key); return list; } private: HashTableType m_table; }; } using AK::HashMap;