diff options
author | Devtools Arcadia <arcadia-devtools@yandex-team.ru> | 2022-02-07 18:08:42 +0300 |
---|---|---|
committer | Devtools Arcadia <arcadia-devtools@mous.vla.yp-c.yandex.net> | 2022-02-07 18:08:42 +0300 |
commit | 1110808a9d39d4b808aef724c861a2e1a38d2a69 (patch) | |
tree | e26c9fed0de5d9873cce7e00bc214573dc2195b7 /util/generic/hash_set.h | |
download | ydb-1110808a9d39d4b808aef724c861a2e1a38d2a69.tar.gz |
intermediate changes
ref:cde9a383711a11544ce7e107a78147fb96cc4029
Diffstat (limited to 'util/generic/hash_set.h')
-rw-r--r-- | util/generic/hash_set.h | 488 |
1 files changed, 488 insertions, 0 deletions
diff --git a/util/generic/hash_set.h b/util/generic/hash_set.h new file mode 100644 index 0000000000..e8088cf23b --- /dev/null +++ b/util/generic/hash_set.h @@ -0,0 +1,488 @@ +#pragma once + +#include "fwd.h" +#include "hash.h" + +#include <initializer_list> +#include <utility> + +#undef value_type + +template <class Value, class HashFcn, class EqualKey, class Alloc> +class THashSet { +private: + using ht = THashTable<Value, Value, HashFcn, ::TIdentity, EqualKey, Alloc>; + ht rep; + + using mutable_iterator = typename ht::iterator; + +public: + using key_type = typename ht::key_type; + using value_type = typename ht::value_type; + using hasher = typename ht::hasher; + using key_equal = typename ht::key_equal; + using allocator_type = typename ht::allocator_type; + using node_allocator_type = typename ht::node_allocator_type; + + using size_type = typename ht::size_type; + using difference_type = typename ht::difference_type; + using pointer = typename ht::const_pointer; + using const_pointer = typename ht::const_pointer; + using reference = typename ht::const_reference; + using const_reference = typename ht::const_reference; + + using iterator = typename ht::const_iterator; + using const_iterator = typename ht::const_iterator; + using insert_ctx = typename ht::insert_ctx; + + hasher hash_function() const { + return rep.hash_function(); + } + key_equal key_eq() const { + return rep.key_eq(); + } + +public: + THashSet() { + } + template <class TT> + explicit THashSet(TT* allocParam, size_type n = 0) + : rep(n, hasher(), key_equal(), allocParam) + { + } + explicit THashSet(size_type n) + : rep(n, hasher(), key_equal()) + { + } + THashSet(size_type n, const hasher& hf) + : rep(n, hf, key_equal()) + { + } + THashSet(size_type n, const hasher& hf, const key_equal& eql) + : rep(n, hf, eql) + { + } + + THashSet(std::initializer_list<value_type> list) + : rep(list.size(), hasher(), key_equal()) + { + rep.insert_unique(list.begin(), list.end()); + } + THashSet(std::initializer_list<value_type> list, size_type n) + : rep(n, hasher(), key_equal()) + { + rep.insert_unique(list.begin(), list.end()); + } + THashSet(std::initializer_list<value_type> list, size_type n, const hasher& hf) + : rep(n, hf, key_equal()) + { + rep.insert_unique(list.begin(), list.end()); + } + THashSet(std::initializer_list<value_type> list, size_type n, const hasher& hf, const key_equal& eql) + : rep(n, hf, eql) + { + rep.insert_unique(list.begin(), list.end()); + } + + template <class InputIterator> + THashSet(InputIterator f, InputIterator l) + : rep(0, hasher(), key_equal()) + { + rep.insert_unique(f, l); + } + template <class InputIterator> + THashSet(InputIterator f, InputIterator l, size_type n) + : rep(n, hasher(), key_equal()) + { + rep.insert_unique(f, l); + } + template <class InputIterator> + THashSet(InputIterator f, InputIterator l, size_type n, + const hasher& hf) + : rep(n, hf, key_equal()) + { + rep.insert_unique(f, l); + } + template <class InputIterator> + THashSet(InputIterator f, InputIterator l, size_type n, + const hasher& hf, const key_equal& eql) + : rep(n, hf, eql) + { + rep.insert_unique(f, l); + } + + // THashSet has implicit copy/move constructors and copy-/move-assignment operators + // because its implementation is backed by THashTable. + // See hash_ut.cpp + +public: + size_type size() const { + return rep.size(); + } + size_type max_size() const { + return rep.max_size(); + } + + Y_PURE_FUNCTION bool empty() const { + return rep.empty(); + } + explicit operator bool() const noexcept { + return !empty(); + } + void swap(THashSet& hs) { + rep.swap(hs.rep); + } + + iterator begin() const { + return rep.begin(); + } + iterator end() const { + return rep.end(); + } + iterator cbegin() const { + return rep.begin(); + } + iterator cend() const { + return rep.end(); + } + +public: + void insert(std::initializer_list<value_type> ilist) { + insert(ilist.begin(), ilist.end()); + } + + template <class InputIterator> + void insert(InputIterator f, InputIterator l) { + rep.insert_unique(f, l); + } + + std::pair<iterator, bool> insert(const value_type& obj) { + std::pair<mutable_iterator, bool> p = rep.insert_unique(obj); + return std::pair<iterator, bool>(p.first, p.second); + } + template <typename... Args> + std::pair<iterator, bool> emplace(Args&&... args) { + std::pair<mutable_iterator, bool> p = rep.emplace_unique(std::forward<Args>(args)...); + return std::pair<iterator, bool>(p.first, p.second); + } + + iterator insert(const_iterator, const value_type& obj) { // insert_hint + std::pair<mutable_iterator, bool> p = rep.insert_unique(obj); + return p.first; + } + + std::pair<iterator, bool> insert_noresize(const value_type& obj) { + std::pair<mutable_iterator, bool> p = rep.insert_unique_noresize(obj); + return std::pair<iterator, bool>(p.first, p.second); + } + template <typename... Args> + std::pair<iterator, bool> emplace_noresize(Args&&... args) { + std::pair<mutable_iterator, bool> p = rep.emplace_unique_noresize(std::forward<Args>(args)...); + return std::pair<iterator, bool>(p.first, p.second); + } + + template <class TheObj> + iterator insert_direct(const TheObj& obj, const insert_ctx& ins) { + return rep.insert_direct(obj, ins); + } + template <typename... Args> + iterator emplace_direct(const insert_ctx& ins, Args&&... args) { + return rep.emplace_direct(ins, std::forward<Args>(args)...); + } + + template <class TheKey> + const_iterator find(const TheKey& key) const { + return rep.find(key); + } + template <class TheKey> + iterator find(const TheKey& key, insert_ctx& ins) { + return rep.find_i(key, ins); + } + + template <class TheKey> + bool contains(const TheKey& key) const { + return rep.find(key) != rep.end(); + } + template <class TheKey> + bool contains(const TheKey& key, insert_ctx& ins) { + return rep.find_i(key, ins) != rep.end(); + } + + template <class TKey> + size_type count(const TKey& key) const { + return rep.count(key); + } + + template <class TKey> + std::pair<iterator, iterator> equal_range(const TKey& key) const { + return rep.equal_range(key); + } + + size_type erase(const key_type& key) { + return rep.erase(key); + } + void erase(iterator it) { + rep.erase(it); + } + void erase(iterator f, iterator l) { + rep.erase(f, l); + } + void clear() { + rep.clear(); + } + void clear(size_t downsize_hint) { + rep.clear(downsize_hint); + } + void basic_clear() { + rep.basic_clear(); + } + void release_nodes() { + rep.release_nodes(); + } + + template <class KeySaver> + int save_for_st(IOutputStream* stream, KeySaver& ks) const { + return rep.template save_for_st<KeySaver>(stream, ks); + } + +public: + void reserve(size_type hint) { + rep.reserve(hint); + } + size_type bucket_count() const { + return rep.bucket_count(); + } + size_type bucket_size(size_type n) const { + return rep.bucket_size(n); + } + node_allocator_type& GetNodeAllocator() { + return rep.GetNodeAllocator(); + } +}; + +template <class Value, class HashFcn, class EqualKey, class Alloc> +inline bool operator==(const THashSet<Value, HashFcn, EqualKey, Alloc>& hs1, const THashSet<Value, HashFcn, EqualKey, Alloc>& hs2) { + if (hs1.size() != hs2.size()) { + return false; + } + for (const auto& it : hs1) { + if (!hs2.contains(it)) { + return false; + } + } + return true; +} + +template <class Value, class HashFcn, class EqualKey, class Alloc> +inline bool operator!=(const THashSet<Value, HashFcn, EqualKey, Alloc>& hs1, const THashSet<Value, HashFcn, EqualKey, Alloc>& hs2) { + return !(hs1 == hs2); +} + +template <class Value, class HashFcn, class EqualKey, class Alloc> +class THashMultiSet { +private: + using ht = THashTable<Value, Value, HashFcn, ::TIdentity, EqualKey, Alloc>; + ht rep; + +public: + using key_type = typename ht::key_type; + using value_type = typename ht::value_type; + using hasher = typename ht::hasher; + using key_equal = typename ht::key_equal; + using allocator_type = typename ht::allocator_type; + using node_allocator_type = typename ht::node_allocator_type; + + using size_type = typename ht::size_type; + using difference_type = typename ht::difference_type; + using pointer = typename ht::const_pointer; + using const_pointer = typename ht::const_pointer; + using reference = typename ht::const_reference; + using const_reference = typename ht::const_reference; + + using iterator = typename ht::const_iterator; + using const_iterator = typename ht::const_iterator; + + hasher hash_function() const { + return rep.hash_function(); + } + key_equal key_eq() const { + return rep.key_eq(); + } + +public: + THashMultiSet() + : rep(0, hasher(), key_equal()) + { + } + explicit THashMultiSet(size_type n) + : rep(n, hasher(), key_equal()) + { + } + THashMultiSet(size_type n, const hasher& hf) + : rep(n, hf, key_equal()) + { + } + THashMultiSet(size_type n, const hasher& hf, const key_equal& eql) + : rep(n, hf, eql) + { + } + + template <class InputIterator> + THashMultiSet(InputIterator f, InputIterator l) + : rep(0, hasher(), key_equal()) + { + rep.insert_equal(f, l); + } + template <class InputIterator> + THashMultiSet(InputIterator f, InputIterator l, size_type n) + : rep(n, hasher(), key_equal()) + { + rep.insert_equal(f, l); + } + template <class InputIterator> + THashMultiSet(InputIterator f, InputIterator l, size_type n, + const hasher& hf) + : rep(n, hf, key_equal()) + { + rep.insert_equal(f, l); + } + template <class InputIterator> + THashMultiSet(InputIterator f, InputIterator l, size_type n, + const hasher& hf, const key_equal& eql) + : rep(n, hf, eql) + { + rep.insert_equal(f, l); + } + + THashMultiSet(std::initializer_list<value_type> list) + : rep(list.size(), hasher(), key_equal()) + { + rep.insert_equal(list.begin(), list.end()); + } + + // THashMultiSet has implicit copy/move constructors and copy-/move-assignment operators + // because its implementation is backed by THashTable. + // See hash_ut.cpp + +public: + size_type size() const { + return rep.size(); + } + size_type max_size() const { + return rep.max_size(); + } + + Y_PURE_FUNCTION bool empty() const { + return rep.empty(); + } + explicit operator bool() const noexcept { + return !empty(); + } + void swap(THashMultiSet& hs) { + rep.swap(hs.rep); + } + + iterator begin() const { + return rep.begin(); + } + iterator end() const { + return rep.end(); + } + iterator cbegin() const { + return rep.begin(); + } + iterator cend() const { + return rep.end(); + } + +public: + iterator insert(const value_type& obj) { + return rep.insert_equal(obj); + } + template <typename... Args> + iterator emplace(Args&&... args) { + return rep.emplace_equal(std::forward<Args>(args)...); + } + template <class InputIterator> + void insert(InputIterator f, InputIterator l) { + rep.insert_equal(f, l); + } + iterator insert_noresize(const value_type& obj) { + return rep.insert_equal_noresize(obj); + } + + template <class TKey> + iterator find(const TKey& key) const { + return rep.find(key); + } + + template <class TKey> + size_type count(const TKey& key) const { + return rep.count(key); + } + + template <class TKey> + std::pair<iterator, iterator> equal_range(const TKey& key) const { + return rep.equal_range(key); + } + + size_type erase(const key_type& key) { + return rep.erase(key); + } + void erase(iterator it) { + rep.erase(it); + } + void erase(iterator f, iterator l) { + rep.erase(f, l); + } + void clear() { + rep.clear(); + } + void clear(size_t downsize_hint) { + rep.clear(downsize_hint); + } + void basic_clear() { + rep.basic_clear(); + } + void release_nodes() { + rep.release_nodes(); + } + +public: + void reserve(size_type hint) { + rep.reserve(hint); + } + size_type bucket_count() const { + return rep.bucket_count(); + } + size_type bucket_size(size_type n) const { + return rep.bucket_size(n); + } + node_allocator_type& GetNodeAllocator() { + return rep.GetNodeAllocator(); + } +}; + +template <class Val, class HashFcn, class EqualKey, class Alloc> +inline bool operator==(const THashMultiSet<Val, HashFcn, EqualKey, Alloc>& hs1, const THashMultiSet<Val, HashFcn, EqualKey, Alloc>& hs2) { + if (hs1.size() != hs2.size()) { + return false; + } + EqualKey equalKey; + auto it = hs1.begin(); + while (it != hs1.end()) { + const auto& value = *it; + size_t count = 0; + for (; (it != hs1.end()) && (equalKey(*it, value)); ++it, ++count) { + } + if (hs2.count(value) != count) { + return false; + } + } + return true; +} + +template <class Val, class HashFcn, class EqualKey, class Alloc> +inline bool operator!=(const THashMultiSet<Val, HashFcn, EqualKey, Alloc>& hs1, const THashMultiSet<Val, HashFcn, EqualKey, Alloc>& hs2) { + return !(hs1 == hs2); +} |