#pragma once #include #include #include #include namespace AK { template class HashMap { private: struct Entry { K key; V value; bool operator==(const Entry& other) { return key == other.key; } }; struct EntryTraits { static unsigned hash(const Entry& entry) { return Traits::hash(entry.key); } static void dump(const Entry& entry) { kprintf("key="); Traits::dump(entry.key); kprintf(" value="); Traits::dump(entry.value); } }; public: HashMap() {} HashMap(HashMap&& other) : m_table(move(other.m_table)) { } HashMap& operator=(HashMap&& other) { if (this != &other) { m_table = move(other.m_table); } return *this; } 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&, const V&); void set(const K&, V&&); void remove(const K&); 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&); ConstIteratorType begin() const { return m_table.begin(); } ConstIteratorType end() const { return m_table.end(); } ConstIteratorType find(const K&) const; void ensure_capacity(int capacity) { m_table.ensure_capacity(capacity); } void dump() const { m_table.dump(); } V get(const K& key) const { auto it = find(key); if (it == end()) return V(); 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: HashTable m_table; }; template void HashMap::set(const K& key, V&& value) { m_table.set(Entry { key, move(value) }); } template void HashMap::set(const K& key, const V& value) { m_table.set(Entry { key, value }); } template void HashMap::remove(const K& key) { Entry dummy { key, V() }; m_table.remove(dummy); } template auto HashMap::find(const K& key) -> IteratorType { Entry dummy { key, V() }; return m_table.find(dummy); } template auto HashMap::find(const K& key) const -> ConstIteratorType { Entry dummy { key, V() }; return m_table.find(dummy); } } using AK::HashMap;