momo  3.10
unordered_set.h
Go to the documentation of this file.
1 /**********************************************************\
2 
3  This file is part of the
4  https://github.com/morzhovets/momo
5  project, distributed under the MIT License. See
6  https://github.com/morzhovets/momo/blob/branch_cpp11/LICENSE
7  for details.
8 
9  momo/stdish/unordered_set.h
10 
11  namespace momo::stdish:
12  class unordered_set
13  class unordered_set_open
14 
15 \**********************************************************/
16 
17 #ifndef MOMO_INCLUDE_GUARD_STDISH_UNORDERED_SET
18 #define MOMO_INCLUDE_GUARD_STDISH_UNORDERED_SET
19 
20 #include "../HashSet.h"
21 #include "set_map_utility.h"
22 
23 namespace momo
24 {
25 
26 namespace stdish
27 {
28 
51 template<typename TKey,
52  typename THashFunc = HashCoder<TKey>,
53  typename TEqualFunc = std::equal_to<TKey>,
54  typename TAllocator = std::allocator<TKey>,
55  typename THashSet = HashSet<TKey, HashTraitsStd<TKey, THashFunc, TEqualFunc>,
56  MemManagerStd<TAllocator>>>
58 {
59 private:
60  typedef THashSet HashSet;
61  typedef typename HashSet::HashTraits HashTraits;
62  typedef typename HashSet::MemManager MemManager;
63 
64 public:
65  typedef TKey key_type;
66  typedef THashFunc hasher;
67  typedef TEqualFunc key_equal;
68  typedef TAllocator allocator_type;
69 
71 
72  typedef size_t size_type;
73  typedef ptrdiff_t difference_type;
74 
76 
78  typedef typename HashSet::Iterator iterator;
79 
80  //typedef typename iterator::Reference reference;
83 
84  //typedef typename iterator::Pointer pointer;
85  typedef value_type* pointer;
87  //typedef typename std::allocator_traits<allocator_type>::pointer pointer;
88  //typedef typename std::allocator_traits<allocator_type>::const_pointer const_pointer;
89 
92 
95 
96 private:
97  template<typename KeyArg>
98  struct IsValidKeyArg : public HashTraits::template IsValidKeyArg<KeyArg>
99  {
100  };
101 
102  struct NodeTypeProxy : private node_type
103  {
104  typedef node_type NodeType;
105  MOMO_DECLARE_PROXY_FUNCTION(NodeType, GetExtractedItem,
106  typename NodeType::SetExtractedItem&)
107  };
108 
109 public:
111  {
112  }
113 
114  explicit unordered_set(const allocator_type& alloc)
115  : mHashSet(HashTraits(), MemManager(alloc))
116  {
117  }
118 
119  explicit unordered_set(size_type bucketCount, const allocator_type& alloc = allocator_type())
120  : mHashSet(HashTraits(bucketCount), MemManager(alloc))
121  {
122  }
123 
124  unordered_set(size_type bucketCount, const hasher& hashFunc,
125  const allocator_type& alloc = allocator_type())
126  : mHashSet(HashTraits(bucketCount, hashFunc), MemManager(alloc))
127  {
128  }
129 
130  unordered_set(size_type bucketCount, const hasher& hashFunc, const key_equal& equalFunc,
131  const allocator_type& alloc = allocator_type())
132  : mHashSet(HashTraits(bucketCount, hashFunc, equalFunc), MemManager(alloc))
133  {
134  }
135 
136  template<typename Iterator>
137  unordered_set(Iterator first, Iterator last)
138  {
139  insert(first, last);
140  }
141 
142  template<typename Iterator>
143  unordered_set(Iterator first, Iterator last, size_type bucketCount,
144  const allocator_type& alloc = allocator_type())
145  : unordered_set(bucketCount, alloc)
146  {
147  insert(first, last);
148  }
149 
150  template<typename Iterator>
151  unordered_set(Iterator first, Iterator last, size_type bucketCount, const hasher& hashFunc,
152  const allocator_type& alloc = allocator_type())
153  : unordered_set(bucketCount, hashFunc, alloc)
154  {
155  insert(first, last);
156  }
157 
158  template<typename Iterator>
159  unordered_set(Iterator first, Iterator last, size_type bucketCount, const hasher& hashFunc,
160  const key_equal& equalFunc, const allocator_type& alloc = allocator_type())
161  : unordered_set(bucketCount, hashFunc, equalFunc, alloc)
162  {
163  insert(first, last);
164  }
165 
167  : mHashSet(values)
168  {
169  }
170 
172  size_type bucketCount, const allocator_type& alloc = allocator_type())
173  : mHashSet(values, HashTraits(bucketCount), MemManager(alloc))
174  {
175  }
176 
178  size_type bucketCount, const hasher& hashFunc, const allocator_type& alloc = allocator_type())
179  : mHashSet(values, HashTraits(bucketCount, hashFunc), MemManager(alloc))
180  {
181  }
182 
184  size_type bucketCount, const hasher& hashFunc, const key_equal& equalFunc,
185  const allocator_type& alloc = allocator_type())
186  : mHashSet(values, HashTraits(bucketCount, hashFunc, equalFunc), MemManager(alloc))
187  {
188  }
189 
190 #ifdef MOMO_HAS_CONTAINERS_RANGES
191  template<std::ranges::input_range Range>
192  requires std::convertible_to<std::ranges::range_reference_t<Range>, value_type>
193  unordered_set(std::from_range_t, Range&& values)
194  {
195  insert_range(std::forward<Range>(values));
196  }
197 
198  template<std::ranges::input_range Range>
199  requires std::convertible_to<std::ranges::range_reference_t<Range>, value_type>
200  unordered_set(std::from_range_t, Range&& values, size_type bucketCount,
201  const allocator_type& alloc = allocator_type())
202  : unordered_set(bucketCount, alloc)
203  {
204  insert_range(std::forward<Range>(values));
205  }
206 
207  template<std::ranges::input_range Range>
208  requires std::convertible_to<std::ranges::range_reference_t<Range>, value_type>
209  unordered_set(std::from_range_t, Range&& values, size_type bucketCount, const hasher& hashFunc,
210  const allocator_type& alloc = allocator_type())
211  : unordered_set(bucketCount, hashFunc, alloc)
212  {
213  insert_range(std::forward<Range>(values));
214  }
215 
216  template<std::ranges::input_range Range>
217  requires std::convertible_to<std::ranges::range_reference_t<Range>, value_type>
218  unordered_set(std::from_range_t, Range&& values, size_type bucketCount, const hasher& hashFunc,
219  const key_equal& equalFunc, const allocator_type& alloc = allocator_type())
220  : unordered_set(bucketCount, hashFunc, equalFunc, alloc)
221  {
222  insert_range(std::forward<Range>(values));
223  }
224 #endif // MOMO_HAS_CONTAINERS_RANGES
225 
226  unordered_set(unordered_set&& right) noexcept
227  : mHashSet(std::move(right.mHashSet))
228  {
229  }
230 
232  noexcept(std::is_empty<allocator_type>::value)
233  : mHashSet(pvCreateSet(std::move(right), alloc))
234  {
235  }
236 
238  : mHashSet(right.mHashSet)
239  {
240  }
241 
243  : mHashSet(right.mHashSet, MemManager(alloc))
244  {
245  }
246 
247  ~unordered_set() = default;
248 
250  noexcept(std::is_empty<allocator_type>::value ||
251  std::allocator_traits<allocator_type>::propagate_on_container_move_assignment::value)
252  {
253  if (this != &right)
254  {
255  bool propagate = std::is_empty<allocator_type>::value ||
256  std::allocator_traits<allocator_type>::propagate_on_container_move_assignment::value;
257  allocator_type alloc = (propagate ? &right : this)->get_allocator();
258  mHashSet = pvCreateSet(std::move(right), alloc);
259  }
260  return *this;
261  }
262 
264  {
265  if (this != &right)
266  {
267  bool propagate = std::is_empty<allocator_type>::value ||
268  std::allocator_traits<allocator_type>::propagate_on_container_copy_assignment::value;
269  allocator_type alloc = (propagate ? &right : this)->get_allocator();
270  mHashSet = HashSet(right.mHashSet, MemManager(alloc));
271  }
272  return *this;
273  }
274 
275  unordered_set& operator=(std::initializer_list<value_type> values)
276  {
277  mHashSet = HashSet(values, mHashSet.GetHashTraits(), MemManager(get_allocator()));
278  return *this;
279  }
280 
281  void swap(unordered_set& right) noexcept
282  {
283  MOMO_ASSERT(std::allocator_traits<allocator_type>::propagate_on_container_swap::value
284  || get_allocator() == right.get_allocator());
285  mHashSet.Swap(right.mHashSet);
286  }
287 
288  friend void swap(unordered_set& left, unordered_set& right) noexcept
289  {
290  left.swap(right);
291  }
292 
294  {
295  return mHashSet;
296  }
297 
299  {
300  return mHashSet;
301  }
302 
303  const_iterator begin() const noexcept
304  {
305  return mHashSet.GetBegin();
306  }
307 
308  //iterator begin() noexcept
309 
310  const_iterator end() const noexcept
311  {
312  return mHashSet.GetEnd();
313  }
314 
315  //iterator end() noexcept
316 
317  const_iterator cbegin() const noexcept
318  {
319  return begin();
320  }
321 
322  const_iterator cend() const noexcept
323  {
324  return end();
325  }
326 
327  float max_load_factor() const noexcept
328  {
329  return mHashSet.GetHashTraits().GetMaxLoadFactor(HashSet::bucketMaxItemCount);
330  }
331 
332  void max_load_factor(float maxLoadFactor)
333  {
334  if (maxLoadFactor == max_load_factor())
335  return;
336  if (maxLoadFactor <= 0.0 || maxLoadFactor > static_cast<float>(HashSet::bucketMaxItemCount))
337  throw std::out_of_range("invalid load factor");
338  HashTraits hashTraits(mHashSet.GetHashTraits(), maxLoadFactor);
339  HashSet hashSet(hashTraits, MemManager(get_allocator()));
340  hashSet.Reserve(size());
341  hashSet.Insert(begin(), end());
342  mHashSet = std::move(hashSet);
343  }
344 
346  {
347  return mHashSet.GetHashTraits().GetHashFunc();
348  }
349 
351  {
352  return mHashSet.GetHashTraits().GetEqualFunc();
353  }
354 
355  allocator_type get_allocator() const noexcept
356  {
357  return allocator_type(mHashSet.GetMemManager().GetByteAllocator());
358  }
359 
360  size_type max_size() const noexcept
361  {
362  return std::allocator_traits<allocator_type>::max_size(get_allocator());
363  }
364 
365  size_type size() const noexcept
366  {
367  return mHashSet.GetCount();
368  }
369 
370  MOMO_NODISCARD bool empty() const noexcept
371  {
372  return mHashSet.IsEmpty();
373  }
374 
375  void clear() noexcept
376  {
377  mHashSet.Clear();
378  }
379 
380  void rehash(size_type bucketCount)
381  {
382  bucketCount = std::minmax(bucketCount, size_t{2}).second;
383  size_t logBucketCount = momo::internal::UIntMath<>::Log2(bucketCount - 1) + 1;
384  bucketCount = size_t{1} << logBucketCount;
385  reserve(mHashSet.GetHashTraits().CalcCapacity(bucketCount, HashSet::bucketMaxItemCount));
386  }
387 
389  {
390  mHashSet.Reserve(count);
391  }
392 
394  {
395  return mHashSet.Find(key);
396  }
397 
398  //iterator find(const key_type& key)
399 
400  template<typename KeyArg>
402  const_iterator> find(const KeyArg& key) const
403  {
404  return mHashSet.Find(key);
405  }
406 
407  //template<typename KeyArg>
408  //momo::internal::EnableIf<IsValidKeyArg<KeyArg>::value,
409  //iterator> find(const KeyArg& key)
410 
412  {
413  return contains(key) ? 1 : 0;
414  }
415 
416  template<typename KeyArg>
418  size_type> count(const KeyArg& key) const
419  {
420  return contains(key) ? 1 : 0;
421  }
422 
423  MOMO_FORCEINLINE bool contains(const key_type& key) const
424  {
425  return mHashSet.ContainsKey(key);
426  }
427 
428  template<typename KeyArg>
430  bool> contains(const KeyArg& key) const
431  {
432  return mHashSet.ContainsKey(key);
433  }
434 
435  MOMO_FORCEINLINE std::pair<const_iterator, const_iterator> equal_range(
436  const key_type& key) const
437  {
438  return { find(key), end() };
439  }
440 
441  //std::pair<iterator, iterator> equal_range(const key_type& key)
442 
443  template<typename KeyArg>
445  std::pair<const_iterator, const_iterator>> equal_range(const KeyArg& key) const
446  {
447  return { find(key), end() };
448  }
449 
450  //template<typename KeyArg>
451  //momo::internal::EnableIf<IsValidKeyArg<KeyArg>::value,
452  //std::pair<iterator, iterator>> equal_range(const KeyArg& key)
453 
454  std::pair<iterator, bool> insert(value_type&& value)
455  {
456  typename HashSet::InsertResult res = mHashSet.Insert(std::move(value));
457  return { res.position, res.inserted };
458  }
459 
461  {
462 #ifdef MOMO_USE_UNORDERED_HINT_ITERATORS
463  return mHashSet.Add(hint, std::move(value));
464 #else
465  (void)hint;
466  return insert(std::move(value)).first;
467 #endif
468  }
469 
470  std::pair<iterator, bool> insert(const value_type& value)
471  {
472  typename HashSet::InsertResult res = mHashSet.Insert(value);
473  return { res.position, res.inserted };
474  }
475 
477  {
478 #ifdef MOMO_USE_UNORDERED_HINT_ITERATORS
479  return mHashSet.Add(hint, value);
480 #else
481  (void)hint;
482  return insert(value).first;
483 #endif
484  }
485 
487  {
488  if (node.empty())
489  return { end(), false, node_type() };
490  typename HashSet::InsertResult res = mHashSet.Insert(
491  std::move(NodeTypeProxy::GetExtractedItem(node)));
492  return { res.position, res.inserted, res.inserted ? node_type() : std::move(node) };
493  }
494 
496  {
497 #ifdef MOMO_USE_UNORDERED_HINT_ITERATORS
498  if (node.empty())
499  return end();
500  return mHashSet.Add(hint, std::move(NodeTypeProxy::GetExtractedItem(node)));
501 #else
502  (void)hint;
503  return insert(std::move(node)).position;
504 #endif
505  }
506 
507  template<typename Iterator>
508  void insert(Iterator first, Iterator last)
509  {
510  pvInsertRange(first, last);
511  }
512 
513  void insert(std::initializer_list<value_type> values)
514  {
515  mHashSet.Insert(values);
516  }
517 
518 #ifdef MOMO_HAS_CONTAINERS_RANGES
519  template<std::ranges::input_range Range>
520  requires std::convertible_to<std::ranges::range_reference_t<Range>, value_type>
521  void insert_range(Range&& values)
522  {
523  pvInsertRange(std::ranges::begin(values), std::ranges::end(values));
524  }
525 #endif // MOMO_HAS_CONTAINERS_RANGES
526 
527  template<typename... ValueArgs>
528  std::pair<iterator, bool> emplace(ValueArgs&&... valueArgs)
529  {
530  MemManager& memManager = mHashSet.GetMemManager();
531  typename HashSet::ExtractedItem extItem;
532  typedef typename HashSet::ItemTraits::template Creator<ValueArgs...> ValueCreator;
533  extItem.Create(ValueCreator(memManager, std::forward<ValueArgs>(valueArgs)...));
534  typename HashSet::InsertResult res = mHashSet.Insert(std::move(extItem));
535  return { res.position, res.inserted };
536  }
537 
538  template<typename ValueArg>
540  std::pair<iterator, bool>> emplace(ValueArg&& valueArg)
541  {
542  typename HashSet::InsertResult res = mHashSet.InsertVar(
543  static_cast<const key_type&>(valueArg), std::forward<ValueArg>(valueArg));
544  return { res.position, res.inserted };
545  }
546 
547  template<typename... ValueArgs>
548  iterator emplace_hint(const_iterator hint, ValueArgs&&... valueArgs)
549  {
550 #ifdef MOMO_USE_UNORDERED_HINT_ITERATORS
551  return mHashSet.AddVar(hint, std::forward<ValueArgs>(valueArgs)...);
552 #else
553  (void)hint;
554  return emplace(std::forward<ValueArgs>(valueArgs)...).first;
555 #endif
556  }
557 
559  {
560  return mHashSet.Remove(where);
561  }
562 
563  //iterator erase(iterator where)
564 
566  {
567  if (first == begin() && last == end())
568  {
569  clear();
570  return end();
571  }
572  if (first == last)
573  return first;
574  if (first != end() && std::next(first) == last)
575  return erase(first);
576  throw std::invalid_argument("invalid unordered_set erase arguments");
577  }
578 
580  {
581  return mHashSet.Remove(key) ? 1 : 0;
582  }
583 
584  template<typename ValueFilter>
585  friend size_type erase_if(unordered_set& cont, const ValueFilter& valueFilter)
586  {
587  return cont.mHashSet.Remove(valueFilter);
588  }
589 
591  {
592  return node_type(*this, where); // need RVO for exception safety
593  }
594 
596  {
597  const_iterator iter = find(key);
598  return (iter != end()) ? extract(iter) : node_type();
599  }
600 
601  template<typename Set>
602  void merge(Set&& set)
603  {
604  mHashSet.MergeFrom(set.get_nested_container());
605  }
606 
607  size_type max_bucket_count() const noexcept
608  {
610  //return momo::internal::HashSetBuckets<Bucket>::maxBucketCount;
611  }
612 
613  size_type bucket_count() const noexcept
614  {
615  return mHashSet.GetBucketCount();
616  }
617 
618  size_type bucket_size(size_type bucketIndex) const
619  {
620  return mHashSet.GetBucketBounds(bucketIndex).GetCount();
621  }
622 
624  {
625  return mHashSet.GetBucketBounds(bucketIndex).GetBegin();
626  }
627 
629  {
630  return mHashSet.GetBucketBounds(bucketIndex).GetBegin();
631  }
632 
634  {
635  return mHashSet.GetBucketBounds(bucketIndex).GetEnd();
636  }
637 
638  const_local_iterator end(size_type bucketIndex) const
639  {
640  return mHashSet.GetBucketBounds(bucketIndex).GetEnd();
641  }
642 
644  {
645  return begin(bucketIndex);
646  }
647 
649  {
650  return end(bucketIndex);
651  }
652 
653  size_type bucket(const key_type& key) const
654  {
655  return mHashSet.GetBucketIndex(key);
656  }
657 
658  float load_factor() const noexcept
659  {
660  size_t count = size();
661  size_t bucketCount = bucket_count();
662  if (count == 0 && bucketCount == 0)
663  return 0.0;
664  return static_cast<float>(count) / static_cast<float>(bucketCount);
665  }
666 
667  friend bool operator==(const unordered_set& left, const unordered_set& right)
668  {
669  if (left.size() != right.size())
670  return false;
671  for (const_reference ref : left)
672  {
673  if (right.find(ref) == right.end())
674  return false;
675  }
676  return true;
677  }
678 
679  friend bool operator!=(const unordered_set& left, const unordered_set& right)
680  {
681  return !(left == right);
682  }
683 
684 private:
685  static HashSet pvCreateSet(unordered_set&& right, const allocator_type& alloc)
686  {
687  if (right.get_allocator() == alloc)
688  return std::move(right.mHashSet);
689  HashSet hashSet(right.mHashSet.GetHashTraits(), MemManager(alloc));
690  hashSet.MergeFrom(right.mHashSet);
691  return hashSet;
692  }
693 
694  template<typename Iterator, typename Sentinel>
696  void> pvInsertRange(Iterator begin, Sentinel end)
697  {
698  mHashSet.Insert(std::move(begin), std::move(end));
699  }
700 
701  template<typename Iterator, typename Sentinel>
703  void> pvInsertRange(Iterator begin, Sentinel end)
704  {
705  for (Iterator iter = std::move(begin); iter != end; ++iter)
706  emplace(*iter);
707  }
708 
709 private:
710  HashSet mHashSet;
711 };
712 
722 template<typename TKey,
723  typename THashFunc = HashCoder<TKey>,
724  typename TEqualFunc = std::equal_to<TKey>,
725  typename TAllocator = std::allocator<TKey>>
726 class unordered_set_open : public unordered_set<TKey, THashFunc, TEqualFunc, TAllocator,
727  HashSet<TKey, HashTraitsStd<TKey, THashFunc, TEqualFunc, HashBucketOpenDefault>,
728  MemManagerStd<TAllocator>>>
729 {
730 private:
731  typedef unordered_set<TKey, THashFunc, TEqualFunc, TAllocator,
734 
735 public:
736  using typename UnorderedSet::size_type;
737  using typename UnorderedSet::value_type;
738 
739 public:
740  using UnorderedSet::UnorderedSet;
741 
742  unordered_set_open() {} // clang 3.6
743 
744  unordered_set_open& operator=(std::initializer_list<value_type> values)
745  {
746  UnorderedSet::operator=(values);
747  return *this;
748  }
749 
750  friend void swap(unordered_set_open& left, unordered_set_open& right) noexcept
751  {
752  left.swap(right);
753  }
754 
755  template<typename ValueFilter>
756  friend size_type erase_if(unordered_set_open& cont, const ValueFilter& valueFilter)
757  {
758  return cont.get_nested_container().Remove(valueFilter);
759  }
760 };
761 
762 #ifdef MOMO_HAS_DEDUCTION_GUIDES
763 
764 #define MOMO_DECLARE_DEDUCTION_GUIDES(unordered_set) \
765 template<typename Iterator, \
766  typename Key = typename std::iterator_traits<Iterator>::value_type> \
767 unordered_set(Iterator, Iterator) \
768  -> unordered_set<Key>; \
769 template<typename Iterator, \
770  typename Key = typename std::iterator_traits<Iterator>::value_type, \
771  typename Allocator = std::allocator<Key>, \
772  typename = internal::unordered_checker<Key, Allocator, HashCoder<Key>>> \
773 unordered_set(Iterator, Iterator, size_t, Allocator = Allocator()) \
774  -> unordered_set<Key, HashCoder<Key>, std::equal_to<Key>, Allocator>; \
775 template<typename Iterator, typename HashFunc, \
776  typename Key = typename std::iterator_traits<Iterator>::value_type, \
777  typename Allocator = std::allocator<Key>, \
778  typename = internal::unordered_checker<Key, Allocator, HashFunc>> \
779 unordered_set(Iterator, Iterator, size_t, HashFunc, Allocator = Allocator()) \
780  -> unordered_set<Key, HashFunc, std::equal_to<Key>, Allocator>; \
781 template<typename Iterator, typename HashFunc, typename EqualFunc, \
782  typename Key = typename std::iterator_traits<Iterator>::value_type, \
783  typename Allocator = std::allocator<Key>, \
784  typename = internal::unordered_checker<Key, Allocator, HashFunc, EqualFunc>> \
785 unordered_set(Iterator, Iterator, size_t, HashFunc, EqualFunc, Allocator = Allocator()) \
786  -> unordered_set<Key, HashFunc, EqualFunc, Allocator>; \
787 template<typename Key> \
788 unordered_set(std::initializer_list<Key>) \
789  -> unordered_set<Key>; \
790 template<typename Key, \
791  typename Allocator = std::allocator<Key>, \
792  typename = internal::unordered_checker<Key, Allocator, HashCoder<Key>>> \
793 unordered_set(std::initializer_list<Key>, size_t, Allocator = Allocator()) \
794  -> unordered_set<Key, HashCoder<Key>, std::equal_to<Key>, Allocator>; \
795 template<typename Key, typename HashFunc, \
796  typename Allocator = std::allocator<Key>, \
797  typename = internal::unordered_checker<Key, Allocator, HashFunc>> \
798 unordered_set(std::initializer_list<Key>, size_t, HashFunc, Allocator = Allocator()) \
799  -> unordered_set<Key, HashFunc, std::equal_to<Key>, Allocator>; \
800 template<typename Key, typename HashFunc, typename EqualFunc, \
801  typename Allocator = std::allocator<Key>, \
802  typename = internal::unordered_checker<Key, Allocator, HashFunc, EqualFunc>> \
803 unordered_set(std::initializer_list<Key>, size_t, HashFunc, EqualFunc, Allocator = Allocator()) \
804  -> unordered_set<Key, HashFunc, EqualFunc, Allocator>;
805 
806 MOMO_DECLARE_DEDUCTION_GUIDES(unordered_set)
807 MOMO_DECLARE_DEDUCTION_GUIDES(unordered_set_open)
808 
809 #undef MOMO_DECLARE_DEDUCTION_GUIDES
810 
811 #ifdef MOMO_HAS_CONTAINERS_RANGES
812 
813 #define MOMO_DECLARE_DEDUCTION_GUIDES_RANGES(unordered_set) \
814 template<std::ranges::input_range Range, \
815  typename Key = std::ranges::range_value_t<Range>> \
816 unordered_set(std::from_range_t, Range&&) \
817  -> unordered_set<Key>; \
818 template<std::ranges::input_range Range, \
819  typename Key = std::ranges::range_value_t<Range>, \
820  typename Allocator = std::allocator<Key>, \
821  typename = internal::unordered_checker<Key, Allocator, HashCoder<Key>>> \
822 unordered_set(std::from_range_t, Range&&, size_t, Allocator = Allocator()) \
823  -> unordered_set<Key, HashCoder<Key>, std::equal_to<Key>, Allocator>; \
824 template<std::ranges::input_range Range, typename HashFunc, \
825  typename Key = std::ranges::range_value_t<Range>, \
826  typename Allocator = std::allocator<Key>, \
827  typename = internal::unordered_checker<Key, Allocator, HashFunc>> \
828 unordered_set(std::from_range_t, Range&&, size_t, HashFunc, Allocator = Allocator()) \
829  -> unordered_set<Key, HashFunc, std::equal_to<Key>, Allocator>; \
830 template<std::ranges::input_range Range, typename HashFunc, typename EqualFunc, \
831  typename Key = std::ranges::range_value_t<Range>, \
832  typename Allocator = std::allocator<Key>, \
833  typename = internal::unordered_checker<Key, Allocator, HashFunc, EqualFunc>> \
834 unordered_set(std::from_range_t, Range&&, size_t, HashFunc, EqualFunc, Allocator = Allocator()) \
835  -> unordered_set<Key, HashFunc, EqualFunc, Allocator>;
836 
837 MOMO_DECLARE_DEDUCTION_GUIDES_RANGES(unordered_set)
838 MOMO_DECLARE_DEDUCTION_GUIDES_RANGES(unordered_set_open)
839 
840 #undef MOMO_DECLARE_DEDUCTION_GUIDES_RANGES
841 
842 #endif // MOMO_HAS_CONTAINERS_RANGES
843 
844 #endif // MOMO_HAS_DEDUCTION_GUIDES
845 
846 } // namespace stdish
847 
848 } // namespace momo
849 
850 #endif // MOMO_INCLUDE_GUARD_STDISH_UNORDERED_SET
momo::stdish::unordered_set::unordered_set
unordered_set(size_type bucketCount, const allocator_type &alloc=allocator_type())
Definition: unordered_set.h:119
momo::stdish::unordered_set::unordered_set
unordered_set(Iterator first, Iterator last, size_type bucketCount, const hasher &hashFunc, const key_equal &equalFunc, const allocator_type &alloc=allocator_type())
Definition: unordered_set.h:159
momo::stdish::unordered_set::bucket_count
size_type bucket_count() const noexcept
Definition: unordered_set.h:613
momo::internal::InsertResult::position
Position position
Definition: IteratorUtility.h:168
momo::internal::UIntConst::maxSize
static const size_t maxSize
Definition: Utility.h:447
momo::HashSet::MemManager
TMemManager MemManager
Definition: HashSet.h:470
momo::stdish::unordered_set::erase
iterator erase(const_iterator first, const_iterator last)
Definition: unordered_set.h:565
momo::stdish::unordered_set::difference_type
ptrdiff_t difference_type
Definition: unordered_set.h:73
momo::stdish::unordered_set::insert
iterator insert(const_iterator hint, value_type &&value)
Definition: unordered_set.h:460
momo::stdish::unordered_set::count
MOMO_FORCEINLINE momo::internal::EnableIf< IsValidKeyArg< KeyArg >::value, size_type > count(const KeyArg &key) const
Definition: unordered_set.h:418
momo::stdish::unordered_set::get_nested_container
const nested_container_type & get_nested_container() const noexcept
Definition: unordered_set.h:293
momo::stdish::unordered_set_open::size_type
size_t size_type
Definition: unordered_set.h:72
momo::stdish::unordered_set::operator=
unordered_set & operator=(unordered_set &&right) noexcept(std::is_empty< allocator_type >::value||std::allocator_traits< allocator_type >::propagate_on_container_move_assignment::value)
Definition: unordered_set.h:249
momo::stdish::unordered_set::begin
const_local_iterator begin(size_type bucketIndex) const
Definition: unordered_set.h:628
momo::stdish::unordered_set::begin
const_iterator begin() const noexcept
Definition: unordered_set.h:303
momo::stdish::unordered_set::insert
insert_return_type insert(node_type &&node)
Definition: unordered_set.h:486
momo::stdish::unordered_set::unordered_set
unordered_set(const unordered_set &right, const momo::internal::Identity< allocator_type > &alloc)
Definition: unordered_set.h:242
momo::stdish::unordered_set::max_size
size_type max_size() const noexcept
Definition: unordered_set.h:360
momo::stdish::unordered_set::equal_range
MOMO_FORCEINLINE std::pair< const_iterator, const_iterator > equal_range(const key_type &key) const
Definition: unordered_set.h:435
momo::stdish::unordered_set::operator==
friend bool operator==(const unordered_set &left, const unordered_set &right)
Definition: unordered_set.h:667
momo::stdish::unordered_set::unordered_set
unordered_set(unordered_set &&right) noexcept
Definition: unordered_set.h:226
set_map_utility.h
momo::stdish::unordered_set::insert
iterator insert(const_iterator hint, node_type &&node)
Definition: unordered_set.h:495
momo::stdish::unordered_set::end
const_iterator end() const noexcept
Definition: unordered_set.h:310
momo::stdish::internal::insert_return_type
Definition: set_map_utility.h:194
momo::internal::HashSetConstIterator::Pointer
const Item * Pointer
Definition: HashSet.h:206
momo::stdish::unordered_set::clear
void clear() noexcept
Definition: unordered_set.h:375
momo::internal::UIntMath::Log2
static UInt Log2(UInt value) noexcept
Definition: Utility.h:385
momo::stdish::unordered_set::extract
node_type extract(const_iterator where)
Definition: unordered_set.h:590
momo::stdish::unordered_set::max_bucket_count
size_type max_bucket_count() const noexcept
Definition: unordered_set.h:607
momo::stdish::unordered_set::size_type
size_t size_type
Definition: unordered_set.h:72
momo::stdish::unordered_set::insert
void insert(std::initializer_list< value_type > values)
Definition: unordered_set.h:513
momo::stdish::unordered_set::end
local_iterator end(size_type bucketIndex)
Definition: unordered_set.h:633
momo::stdish::unordered_set::unordered_set
unordered_set(size_type bucketCount, const hasher &hashFunc, const key_equal &equalFunc, const allocator_type &alloc=allocator_type())
Definition: unordered_set.h:130
momo::internal::HashSetConstIterator::Reference
const Item & Reference
Definition: HashSet.h:205
momo::stdish::unordered_set::unordered_set
unordered_set(const allocator_type &alloc)
Definition: unordered_set.h:114
momo::HashSet::Reserve
void Reserve(size_t capacity)
Definition: HashSet.h:709
momo::stdish::unordered_set::load_factor
float load_factor() const noexcept
Definition: unordered_set.h:658
momo::stdish::unordered_set::insert
std::pair< iterator, bool > insert(const value_type &value)
Definition: unordered_set.h:470
momo::stdish::set
momo::stdish::set is similar to std::set, but much more efficient in memory usage....
Definition: set.h:58
momo::MemManagerStd
MemManagerStd uses allocator<unsigned char>::allocate and deallocate
Definition: MemManager.h:177
momo::stdish::unordered_set::unordered_set
unordered_set(std::initializer_list< momo::internal::Identity< value_type >> values, size_type bucketCount, const hasher &hashFunc, const allocator_type &alloc=allocator_type())
Definition: unordered_set.h:177
momo::stdish::unordered_set::erase_if
friend size_type erase_if(unordered_set &cont, const ValueFilter &valueFilter)
Definition: unordered_set.h:585
momo::stdish::unordered_set_open::unordered_set_open
unordered_set_open()
Definition: unordered_set.h:742
momo::stdish::unordered_set::equal_range
MOMO_FORCEINLINE momo::internal::EnableIf< IsValidKeyArg< KeyArg >::value, std::pair< const_iterator, const_iterator > > equal_range(const KeyArg &key) const
Definition: unordered_set.h:445
momo::HashSet::Remove
ConstIterator Remove(ConstIterator iter)
Definition: HashSet.h:849
momo::stdish::unordered_set::insert
void insert(Iterator first, Iterator last)
Definition: unordered_set.h:508
momo::internal::InsertResult
Definition: IteratorUtility.h:136
momo::stdish::unordered_set::unordered_set
unordered_set(Iterator first, Iterator last, size_type bucketCount, const allocator_type &alloc=allocator_type())
Definition: unordered_set.h:143
momo::internal::HashSetConstIterator
Definition: HashSet.h:279
momo::stdish::unordered_set::value_type
key_type value_type
Definition: unordered_set.h:75
momo::stdish::unordered_set::unordered_set
unordered_set(std::initializer_list< momo::internal::Identity< value_type >> values, size_type bucketCount, const hasher &hashFunc, const key_equal &equalFunc, const allocator_type &alloc=allocator_type())
Definition: unordered_set.h:183
momo::stdish::unordered_set::empty
MOMO_NODISCARD bool empty() const noexcept
Definition: unordered_set.h:370
momo::stdish::unordered_set::unordered_set
unordered_set()
Definition: unordered_set.h:110
momo::HashSet::Insert
InsertResult Insert(Item &&item)
Definition: HashSet.h:771
momo::stdish::unordered_set::max_load_factor
void max_load_factor(float maxLoadFactor)
Definition: unordered_set.h:332
momo::stdish::unordered_set::~unordered_set
~unordered_set()=default
momo::stdish::unordered_set::unordered_set
unordered_set(const unordered_set &right)
Definition: unordered_set.h:237
momo::HashSet::HashTraits
THashTraits HashTraits
Definition: HashSet.h:469
momo::internal::EnableIf
typename std::enable_if< value, Type >::type EnableIf
Definition: Utility.h:198
momo::stdish::unordered_set::emplace_hint
iterator emplace_hint(const_iterator hint, ValueArgs &&... valueArgs)
Definition: unordered_set.h:548
momo::stdish::unordered_set::operator=
unordered_set & operator=(std::initializer_list< value_type > values)
Definition: unordered_set.h:275
momo::stdish::unordered_set::allocator_type
TAllocator allocator_type
Definition: unordered_set.h:68
momo::stdish::unordered_set::bucket_size
size_type bucket_size(size_type bucketIndex) const
Definition: unordered_set.h:618
momo::stdish::unordered_set::count
MOMO_FORCEINLINE size_type count(const key_type &key) const
Definition: unordered_set.h:411
momo
Definition: Array.h:26
momo::stdish::unordered_set::key_type
TKey key_type
Definition: unordered_set.h:65
momo::HashCoder< TKey >
momo::stdish::unordered_set::hash_function
hasher hash_function() const
Definition: unordered_set.h:345
momo::stdish::set::get_nested_container
const nested_container_type & get_nested_container() const noexcept
Definition: set.h:237
momo::HashSet::bucketMaxItemCount
static const size_t bucketMaxItemCount
Definition: HashSet.h:504
momo::stdish::unordered_set::end
const_local_iterator end(size_type bucketIndex) const
Definition: unordered_set.h:638
momo::internal::InsertResult::inserted
bool inserted
Definition: IteratorUtility.h:171
momo::internal::Identity
EnableIf< true, Type > Identity
Definition: Utility.h:201
momo::stdish::unordered_set::iterator
HashSet::Iterator iterator
Definition: unordered_set.h:78
momo::stdish::unordered_set::reserve
void reserve(size_type count)
Definition: unordered_set.h:388
momo::stdish::unordered_set::get_allocator
allocator_type get_allocator() const noexcept
Definition: unordered_set.h:355
momo::stdish::unordered_set::reference
value_type & reference
Definition: unordered_set.h:81
momo::stdish::unordered_set::contains
MOMO_FORCEINLINE momo::internal::EnableIf< IsValidKeyArg< KeyArg >::value, bool > contains(const KeyArg &key) const
Definition: unordered_set.h:430
momo::stdish::unordered_set::key_eq
key_equal key_eq() const
Definition: unordered_set.h:350
momo::stdish::unordered_set_open
momo::stdish::unordered_set_open is similar to std::unordered_set, but much more efficient in operati...
Definition: unordered_set.h:729
momo::stdish::internal::set_node_handle::SetExtractedItem
TSetExtractedItem SetExtractedItem
Definition: set_map_utility.h:35
momo::stdish::unordered_set::cbegin
const_local_iterator cbegin(size_type bucketIndex) const
Definition: unordered_set.h:643
momo::HashTraitsStd
Definition: HashTraits.h:196
momo::stdish::unordered_set::const_iterator
HashSet::ConstIterator const_iterator
Definition: unordered_set.h:77
momo::stdish::unordered_set::const_reference
const_iterator::Reference const_reference
Definition: unordered_set.h:82
momo::stdish::unordered_set::emplace
momo::internal::EnableIf< std::is_same< key_type, typename std::decay< ValueArg >::type >::value, std::pair< iterator, bool > > emplace(ValueArg &&valueArg)
Definition: unordered_set.h:540
momo::HashSet< TKey, HashTraitsStd< TKey, HashCoder< TKey >, std::equal_to< TKey >, HashBucketOpenDefault >, MemManagerStd< std::allocator< TKey > > >
momo::stdish::unordered_set::bucket
size_type bucket(const key_type &key) const
Definition: unordered_set.h:653
momo::stdish::unordered_set_open::erase_if
friend size_type erase_if(unordered_set_open &cont, const ValueFilter &valueFilter)
Definition: unordered_set.h:756
momo::stdish::unordered_set::hasher
THashFunc hasher
Definition: unordered_set.h:66
momo::stdish::unordered_set::operator=
unordered_set & operator=(const unordered_set &right)
Definition: unordered_set.h:263
momo::stdish::unordered_set::insert
iterator insert(const_iterator hint, const value_type &value)
Definition: unordered_set.h:476
momo::stdish::unordered_set::merge
void merge(Set &&set)
Definition: unordered_set.h:602
std
Definition: ArrayUtility.h:308
MOMO_FORCEINLINE
#define MOMO_FORCEINLINE
Definition: UserSettings.h:114
momo::stdish::unordered_set::find
MOMO_FORCEINLINE momo::internal::EnableIf< IsValidKeyArg< KeyArg >::value, const_iterator > find(const KeyArg &key) const
Definition: unordered_set.h:402
momo::stdish::unordered_set::key_equal
TEqualFunc key_equal
Definition: unordered_set.h:67
momo::stdish::unordered_set::nested_container_type
HashSet nested_container_type
Definition: unordered_set.h:70
momo::stdish::unordered_set::swap
void swap(unordered_set &right) noexcept
Definition: unordered_set.h:281
momo::internal::SetExtractedItem::Create
void Create(ItemCreator &&itemCreator)
Definition: SetUtility.h:321
momo::stdish::unordered_set::unordered_set
unordered_set(Iterator first, Iterator last, size_type bucketCount, const hasher &hashFunc, const allocator_type &alloc=allocator_type())
Definition: unordered_set.h:151
momo::stdish::unordered_set::unordered_set
unordered_set(std::initializer_list< momo::internal::Identity< value_type >> values)
Definition: unordered_set.h:166
momo::stdish::unordered_set::extract
node_type extract(const key_type &key)
Definition: unordered_set.h:595
momo::stdish::unordered_set::local_iterator
const_local_iterator local_iterator
Definition: unordered_set.h:94
momo::stdish::unordered_set::const_pointer
const_iterator::Pointer const_pointer
Definition: unordered_set.h:86
momo::stdish::unordered_set::unordered_set
unordered_set(size_type bucketCount, const hasher &hashFunc, const allocator_type &alloc=allocator_type())
Definition: unordered_set.h:124
momo::stdish::unordered_set::cbegin
const_iterator cbegin() const noexcept
Definition: unordered_set.h:317
momo::stdish::unordered_set::size
size_type size() const noexcept
Definition: unordered_set.h:365
momo::stdish::internal::set_node_handle
Definition: set_map_utility.h:33
momo::stdish::unordered_set::insert
std::pair< iterator, bool > insert(value_type &&value)
Definition: unordered_set.h:454
momo::stdish::unordered_set::const_local_iterator
HashSet::ConstBucketBounds::Iterator const_local_iterator
Definition: unordered_set.h:93
momo::stdish::unordered_set::cend
const_iterator cend() const noexcept
Definition: unordered_set.h:322
momo::stdish::unordered_set::emplace
std::pair< iterator, bool > emplace(ValueArgs &&... valueArgs)
Definition: unordered_set.h:528
momo::stdish::unordered_set::unordered_set
unordered_set(Iterator first, Iterator last)
Definition: unordered_set.h:137
momo::stdish::unordered_set::rehash
void rehash(size_type bucketCount)
Definition: unordered_set.h:380
momo::stdish::unordered_set::contains
MOMO_FORCEINLINE bool contains(const key_type &key) const
Definition: unordered_set.h:423
momo::stdish::unordered_set::operator!=
friend bool operator!=(const unordered_set &left, const unordered_set &right)
Definition: unordered_set.h:679
momo::stdish::unordered_set
momo::stdish::unordered_set is similar to std::unordered_set, but much more efficient in memory usage...
Definition: unordered_set.h:58
momo::stdish::unordered_set_open::operator=
unordered_set_open & operator=(std::initializer_list< value_type > values)
Definition: unordered_set.h:744
momo::stdish::unordered_set::find
MOMO_FORCEINLINE const_iterator find(const key_type &key) const
Definition: unordered_set.h:393
momo::stdish::unordered_set::erase
size_type erase(const key_type &key)
Definition: unordered_set.h:579
momo::stdish::unordered_set::get_nested_container
nested_container_type & get_nested_container() noexcept
Definition: unordered_set.h:298
momo::stdish::unordered_set::unordered_set
unordered_set(std::initializer_list< momo::internal::Identity< value_type >> values, size_type bucketCount, const allocator_type &alloc=allocator_type())
Definition: unordered_set.h:171
momo::stdish::unordered_set::node_type
internal::set_node_handle< typename HashSet::ExtractedItem > node_type
Definition: unordered_set.h:90
momo::stdish::unordered_set::unordered_set
unordered_set(unordered_set &&right, const momo::internal::Identity< allocator_type > &alloc) noexcept(std::is_empty< allocator_type >::value)
Definition: unordered_set.h:231
MOMO_ASSERT
#define MOMO_ASSERT(expr)
Definition: UserSettings.h:162
momo::stdish::unordered_set::swap
friend void swap(unordered_set &left, unordered_set &right) noexcept
Definition: unordered_set.h:288
MOMO_DECLARE_PROXY_FUNCTION
#define MOMO_DECLARE_PROXY_FUNCTION(Object, Func, Result)
Definition: Utility.h:116
momo::internal::SetExtractedItem
Definition: SetUtility.h:256
momo::stdish::unordered_set::pointer
value_type * pointer
Definition: unordered_set.h:85
momo::stdish::unordered_set::erase
iterator erase(const_iterator where)
Definition: unordered_set.h:558
momo::stdish::unordered_set::insert_return_type
internal::insert_return_type< iterator, node_type > insert_return_type
Definition: unordered_set.h:91
MOMO_NODISCARD
#define MOMO_NODISCARD
Definition: UserSettings.h:203
momo::stdish::unordered_set_open::swap
friend void swap(unordered_set_open &left, unordered_set_open &right) noexcept
Definition: unordered_set.h:750
momo::stdish::unordered_set::max_load_factor
float max_load_factor() const noexcept
Definition: unordered_set.h:327
momo::stdish::unordered_set::begin
local_iterator begin(size_type bucketIndex)
Definition: unordered_set.h:623
momo::stdish::unordered_set::cend
const_local_iterator cend(size_type bucketIndex) const
Definition: unordered_set.h:648