123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490 |
- #pragma once
- #include "fwd.h"
- #include "hash.h"
- #include <util/system/compiler.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_one(key);
- }
- void erase(iterator it) {
- rep.erase(it);
- }
- void erase(iterator f, iterator l) {
- rep.erase(f, l);
- }
- Y_REINITIALIZES_OBJECT void clear() {
- rep.clear();
- }
- Y_REINITIALIZES_OBJECT void clear(size_t downsize_hint) {
- rep.clear(downsize_hint);
- }
- Y_REINITIALIZES_OBJECT 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);
- }
- Y_REINITIALIZES_OBJECT void clear() {
- rep.clear();
- }
- Y_REINITIALIZES_OBJECT void clear(size_t downsize_hint) {
- rep.clear(downsize_hint);
- }
- Y_REINITIALIZES_OBJECT 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);
- }
|