momo  3.10
unordered_multimap.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_multimap.h
10 
11  namespace momo::stdish:
12  class unordered_multimap
13  class unordered_multimap_open
14 
15 \**********************************************************/
16 
17 #ifndef MOMO_INCLUDE_GUARD_STDISH_UNORDERED_MULTIMAP
18 #define MOMO_INCLUDE_GUARD_STDISH_UNORDERED_MULTIMAP
19 
20 #include "../HashMultiMap.h"
21 #include "set_map_utility.h"
22 
23 namespace momo
24 {
25 
26 namespace stdish
27 {
28 
58 template<typename TKey, typename TMapped,
59  typename THashFunc = HashCoder<TKey>,
60  typename TEqualFunc = std::equal_to<TKey>,
61  typename TAllocator = std::allocator<std::pair<const TKey, TMapped>>,
62  typename THashMultiMap = HashMultiMap<TKey, TMapped, HashTraitsStd<TKey, THashFunc, TEqualFunc>,
63  MemManagerStd<TAllocator>>>
65 {
66 private:
67  typedef THashMultiMap HashMultiMap;
68  typedef typename HashMultiMap::HashTraits HashTraits;
69  typedef typename HashMultiMap::MemManager MemManager;
70 
71 public:
72  typedef TKey key_type;
73  typedef TMapped mapped_type;
74  typedef THashFunc hasher;
75  typedef TEqualFunc key_equal;
76  typedef TAllocator allocator_type;
77 
78  typedef HashMultiMap nested_container_type;
79 
80  typedef size_t size_type;
81  typedef ptrdiff_t difference_type;
82 
83  typedef std::pair<const key_type, mapped_type> value_type;
84 
88 
89  typedef typename iterator::Reference reference;
91 
92  typedef typename iterator::Pointer pointer;
94  //typedef typename std::allocator_traits<allocator_type>::pointer pointer;
95  //typedef typename std::allocator_traits<allocator_type>::const_pointer const_pointer;
96 
97  //node_type;
98 
99  //local_iterator;
100  //const_local_iterator;
101 
102 private:
103  template<typename KeyArg>
104  struct IsValidKeyArg : public HashTraits::template IsValidKeyArg<KeyArg>
105  {
106  };
107 
108  struct ConstIteratorProxy : public const_iterator
109  {
110  typedef const_iterator ConstIterator;
111  MOMO_DECLARE_PROXY_CONSTRUCTOR(ConstIterator)
112  MOMO_DECLARE_PROXY_FUNCTION(ConstIterator, GetBaseIterator,
114  };
115 
116  struct IteratorProxy : public iterator
117  {
118  typedef iterator Iterator;
120  };
121 
122 public:
124  {
125  }
126 
127  explicit unordered_multimap(const allocator_type& alloc)
128  : mHashMultiMap(HashTraits(), MemManager(alloc))
129  {
130  }
131 
132  explicit unordered_multimap(size_type bucketCount,
133  const allocator_type& alloc = allocator_type())
134  : mHashMultiMap(HashTraits(bucketCount), MemManager(alloc))
135  {
136  }
137 
138  unordered_multimap(size_type bucketCount, const hasher& hashFunc,
139  const allocator_type& alloc = allocator_type())
140  : mHashMultiMap(HashTraits(bucketCount, hashFunc), MemManager(alloc))
141  {
142  }
143 
144  unordered_multimap(size_type bucketCount, const hasher& hashFunc, const key_equal& equalFunc,
145  const allocator_type& alloc = allocator_type())
146  : mHashMultiMap(HashTraits(bucketCount, hashFunc, equalFunc), MemManager(alloc))
147  {
148  }
149 
150  template<typename Iterator>
151  unordered_multimap(Iterator first, Iterator last)
152  {
153  insert(first, last);
154  }
155 
156  template<typename Iterator>
157  unordered_multimap(Iterator first, Iterator last, size_type bucketCount,
158  const allocator_type& alloc = allocator_type())
159  : unordered_multimap(bucketCount, alloc)
160  {
161  insert(first, last);
162  }
163 
164  template<typename Iterator>
165  unordered_multimap(Iterator first, Iterator last, size_type bucketCount,
166  const hasher& hashFunc, const allocator_type& alloc = allocator_type())
167  : unordered_multimap(bucketCount, hashFunc, alloc)
168  {
169  insert(first, last);
170  }
171 
172  template<typename Iterator>
173  unordered_multimap(Iterator first, Iterator last, size_type bucketCount,
174  const hasher& hashFunc, const key_equal& equalFunc,
175  const allocator_type& alloc = allocator_type())
176  : unordered_multimap(bucketCount, hashFunc, equalFunc, alloc)
177  {
178  insert(first, last);
179  }
180 
182  : unordered_multimap(values.begin(), values.end())
183  {
184  }
185 
187  size_type bucketCount, const allocator_type& alloc = allocator_type())
188  : unordered_multimap(values.begin(), values.end(), bucketCount, alloc)
189  {
190  }
191 
193  size_type bucketCount, const hasher& hashFunc, const allocator_type& alloc = allocator_type())
194  : unordered_multimap(values.begin(), values.end(), bucketCount, hashFunc, alloc)
195  {
196  }
197 
199  size_type bucketCount, const hasher& hashFunc, const key_equal& equalFunc,
200  const allocator_type& alloc = allocator_type())
201  : unordered_multimap(values.begin(), values.end(), bucketCount, hashFunc, equalFunc, alloc)
202  {
203  }
204 
205 #ifdef MOMO_HAS_CONTAINERS_RANGES
206  template<std::ranges::input_range Range>
207  requires std::convertible_to<std::ranges::range_reference_t<Range>, value_type>
208  unordered_multimap(std::from_range_t, Range&& values)
209  {
210  insert_range(std::forward<Range>(values));
211  }
212 
213  template<std::ranges::input_range Range>
214  requires std::convertible_to<std::ranges::range_reference_t<Range>, value_type>
215  unordered_multimap(std::from_range_t, Range&& values, size_type bucketCount,
216  const allocator_type& alloc = allocator_type())
217  : unordered_multimap(bucketCount, alloc)
218  {
219  insert_range(std::forward<Range>(values));
220  }
221 
222  template<std::ranges::input_range Range>
223  requires std::convertible_to<std::ranges::range_reference_t<Range>, value_type>
224  unordered_multimap(std::from_range_t, Range&& values, size_type bucketCount,
225  const hasher& hashFunc, const allocator_type& alloc = allocator_type())
226  : unordered_multimap(bucketCount, hashFunc, alloc)
227  {
228  insert_range(std::forward<Range>(values));
229  }
230 
231  template<std::ranges::input_range Range>
232  requires std::convertible_to<std::ranges::range_reference_t<Range>, value_type>
233  unordered_multimap(std::from_range_t, Range&& values, size_type bucketCount,
234  const hasher& hashFunc, const key_equal& equalFunc,
235  const allocator_type& alloc = allocator_type())
236  : unordered_multimap(bucketCount, hashFunc, equalFunc, alloc)
237  {
238  insert_range(std::forward<Range>(values));
239  }
240 #endif // MOMO_HAS_CONTAINERS_RANGES
241 
243  : mHashMultiMap(std::move(right.mHashMultiMap))
244  {
245  }
246 
249  noexcept(std::is_empty<allocator_type>::value)
250  : mHashMultiMap(pvCreateMultiMap(std::move(right), alloc))
251  {
252  }
253 
255  : mHashMultiMap(right.mHashMultiMap)
256  {
257  }
258 
261  : mHashMultiMap(right.mHashMultiMap, MemManager(alloc))
262  {
263  }
264 
265  ~unordered_multimap() = default;
266 
268  noexcept(std::is_empty<allocator_type>::value ||
269  std::allocator_traits<allocator_type>::propagate_on_container_move_assignment::value)
270  {
271  if (this != &right)
272  {
273  bool propagate = std::is_empty<allocator_type>::value ||
274  std::allocator_traits<allocator_type>::propagate_on_container_move_assignment::value;
275  allocator_type alloc = (propagate ? &right : this)->get_allocator();
276  mHashMultiMap = pvCreateMultiMap(std::move(right), alloc);
277  }
278  return *this;
279  }
280 
282  {
283  if (this != &right)
284  {
285  bool propagate = std::is_empty<allocator_type>::value ||
286  std::allocator_traits<allocator_type>::propagate_on_container_copy_assignment::value;
287  allocator_type alloc = (propagate ? &right : this)->get_allocator();
288  mHashMultiMap = HashMultiMap(right.mHashMultiMap, MemManager(alloc));
289  }
290  return *this;
291  }
292 
293  unordered_multimap& operator=(std::initializer_list<value_type> values)
294  {
295  HashMultiMap hashMultiMap(mHashMultiMap.GetHashTraits(), MemManager(get_allocator()));
296  hashMultiMap.Add(values.begin(), values.end());
297  mHashMultiMap = std::move(hashMultiMap);
298  return *this;
299  }
300 
301  void swap(unordered_multimap& right) noexcept
302  {
303  MOMO_ASSERT(std::allocator_traits<allocator_type>::propagate_on_container_swap::value
304  || get_allocator() == right.get_allocator());
305  mHashMultiMap.Swap(right.mHashMultiMap);
306  }
307 
308  friend void swap(unordered_multimap& left, unordered_multimap& right) noexcept
309  {
310  left.swap(right);
311  }
312 
314  {
315  return mHashMultiMap;
316  }
317 
319  {
320  return mHashMultiMap;
321  }
322 
323  const_iterator begin() const noexcept
324  {
325  return ConstIteratorProxy(mHashMultiMap.GetBegin());
326  }
327 
328  iterator begin() noexcept
329  {
330  return IteratorProxy(mHashMultiMap.GetBegin());
331  }
332 
333  const_iterator end() const noexcept
334  {
335  return ConstIteratorProxy(mHashMultiMap.GetEnd());
336  }
337 
338  iterator end() noexcept
339  {
340  return IteratorProxy(mHashMultiMap.GetEnd());
341  }
342 
343  const_iterator cbegin() const noexcept
344  {
345  return begin();
346  }
347 
348  const_iterator cend() const noexcept
349  {
350  return end();
351  }
352 
353  //float max_load_factor() const noexcept
354  //void max_load_factor(float maxLoadFactor)
355 
357  {
358  return mHashMultiMap.GetHashTraits().GetHashFunc();
359  }
360 
362  {
363  return mHashMultiMap.GetHashTraits().GetEqualFunc();
364  }
365 
366  allocator_type get_allocator() const noexcept
367  {
368  return allocator_type(mHashMultiMap.GetMemManager().GetByteAllocator());
369  }
370 
371  size_type max_size() const noexcept
372  {
373  return std::allocator_traits<allocator_type>::max_size(get_allocator());
374  }
375 
376  size_type size() const noexcept
377  {
378  return mHashMultiMap.GetCount();
379  }
380 
381  MOMO_NODISCARD bool empty() const noexcept
382  {
383  return mHashMultiMap.IsEmpty();
384  }
385 
386  void clear() noexcept
387  {
388  mHashMultiMap.Clear();
389  }
390 
391  //void rehash(size_type bucketCount)
392  //void reserve(size_type count)
393 
395  {
396  return equal_range(key).first;
397  }
398 
400  {
401  return equal_range(key).first;
402  }
403 
404  template<typename KeyArg>
406  const_iterator> find(const KeyArg& key) const
407  {
408  return equal_range(key).first;
409  }
410 
411  template<typename KeyArg>
413  iterator> find(const KeyArg& key)
414  {
415  return equal_range(key).first;
416  }
417 
419  {
420  typename HashMultiMap::ConstKeyIterator keyIter = mHashMultiMap.Find(key);
421  return !!keyIter ? keyIter->GetCount() : 0;
422  }
423 
424  template<typename KeyArg>
426  size_type> count(const KeyArg& key) const
427  {
428  typename HashMultiMap::ConstKeyIterator keyIter = mHashMultiMap.Find(key);
429  return !!keyIter ? keyIter->GetCount() : 0;
430  }
431 
432  MOMO_FORCEINLINE bool contains(const key_type& key) const
433  {
434  return count(key) > 0;
435  }
436 
437  template<typename KeyArg>
439  bool> contains(const KeyArg& key) const
440  {
441  return count(key) > 0;
442  }
443 
444  MOMO_FORCEINLINE std::pair<const_iterator, const_iterator> equal_range(const key_type& key) const
445  {
446  return pvEqualRange<const_iterator, ConstIteratorProxy>(mHashMultiMap,
447  mHashMultiMap.Find(key));
448  }
449 
450  MOMO_FORCEINLINE std::pair<iterator, iterator> equal_range(const key_type& key)
451  {
452  return pvEqualRange<iterator, IteratorProxy>(mHashMultiMap, mHashMultiMap.Find(key));
453  }
454 
455  template<typename KeyArg>
457  std::pair<const_iterator, const_iterator>> equal_range(const KeyArg& key) const
458  {
459  return pvEqualRange<const_iterator, ConstIteratorProxy>(mHashMultiMap,
460  mHashMultiMap.Find(key));
461  }
462 
463  template<typename KeyArg>
465  std::pair<iterator, iterator>> equal_range(const KeyArg& key)
466  {
467  return pvEqualRange<iterator, IteratorProxy>(mHashMultiMap, mHashMultiMap.Find(key));
468  }
469 
470  template<typename ValueArg = std::pair<key_type, mapped_type>>
472  iterator> insert(ValueArg&& valueArg)
473  {
474  return emplace(std::forward<ValueArg>(valueArg));
475  }
476 
477  template<typename ValueArg = std::pair<key_type, mapped_type>>
479  iterator> insert(const_iterator, ValueArg&& valueArg)
480  {
481  return insert(std::forward<ValueArg>(valueArg));
482  }
483 
484  template<typename Iterator>
485  void insert(Iterator first, Iterator last)
486  {
487  pvInsertRange(first, last);
488  }
489 
490  void insert(std::initializer_list<value_type> values)
491  {
492  mHashMultiMap.Add(values.begin(), values.end());
493  }
494 
495 #ifdef MOMO_HAS_CONTAINERS_RANGES
496  template<std::ranges::input_range Range>
497  requires std::convertible_to<std::ranges::range_reference_t<Range>, value_type>
498  void insert_range(Range&& values)
499  {
500  pvInsertRange(std::ranges::begin(values), std::ranges::end(values));
501  }
502 #endif // MOMO_HAS_CONTAINERS_RANGES
503 
505  {
506  return pvEmplace(std::tuple<>(), std::tuple<>());
507  }
508 
510  {
511  return emplace();
512  }
513 
514  template<typename ValueArg>
515  iterator emplace(ValueArg&& valueArg)
516  {
517  return pvEmplace(std::forward_as_tuple(std::get<0>(std::forward<ValueArg>(valueArg))),
518  std::forward_as_tuple(std::get<1>(std::forward<ValueArg>(valueArg))));
519  }
520 
521  template<typename ValueArg>
522  iterator emplace_hint(const_iterator, ValueArg&& valueArg)
523  {
524  return emplace(std::forward<ValueArg>(valueArg));
525  }
526 
527  template<typename KeyArg, typename MappedArg>
528  iterator emplace(KeyArg&& keyArg, MappedArg&& mappedArg)
529  {
530  return pvEmplace(std::forward_as_tuple(std::forward<KeyArg>(keyArg)),
531  std::forward_as_tuple(std::forward<MappedArg>(mappedArg)));
532  }
533 
534  template<typename KeyArg, typename MappedArg>
535  iterator emplace_hint(const_iterator, KeyArg&& keyArg, MappedArg&& mappedArg)
536  {
537  return emplace(std::forward<KeyArg>(keyArg), std::forward<MappedArg>(mappedArg));
538  }
539 
540  template<typename... KeyArgs, typename... MappedArgs>
541  iterator emplace(std::piecewise_construct_t,
542  std::tuple<KeyArgs...> keyArgs, std::tuple<MappedArgs...> mappedArgs)
543  {
544  return pvEmplace(std::move(keyArgs), std::move(mappedArgs));
545  }
546 
547  template<typename... KeyArgs, typename... MappedArgs>
548  iterator emplace_hint(const_iterator, std::piecewise_construct_t,
549  std::tuple<KeyArgs...> keyArgs, std::tuple<MappedArgs...> mappedArgs)
550  {
551  return pvEmplace(std::move(keyArgs), std::move(mappedArgs));
552  }
553 
555  {
556  typename HashMultiMap::ConstIterator iter = ConstIteratorProxy::GetBaseIterator(where);
557  typename HashMultiMap::ConstKeyIterator keyIter = iter.GetKeyIterator();
558  if (keyIter->GetCount() == 1)
559  return IteratorProxy(mHashMultiMap.MakeIterator(mHashMultiMap.RemoveKey(keyIter)));
560  else
561  return IteratorProxy(mHashMultiMap.Remove(iter));
562  }
563 
565  {
566  return erase(static_cast<const_iterator>(where));
567  }
568 
570  {
571  if (first == begin() && last == end())
572  {
573  clear();
574  return end();
575  }
576  if (first == last)
577  {
578  return IteratorProxy(mHashMultiMap.MakeMutableIterator(
579  ConstIteratorProxy::GetBaseIterator(first)));
580  }
581  if (first != end())
582  {
583  if (std::next(first) == last)
584  return erase(first);
585  typename HashMultiMap::ConstKeyIterator keyIter =
586  ConstIteratorProxy::GetBaseIterator(first).GetKeyIterator();
587  if (last == ConstIteratorProxy(mHashMultiMap.MakeIterator(keyIter, keyIter->GetCount())))
588  return IteratorProxy(mHashMultiMap.MakeIterator(mHashMultiMap.RemoveKey(keyIter)));
589  }
590  throw std::invalid_argument("invalid unordered_multimap erase arguments");
591  }
592 
594  {
595  return mHashMultiMap.RemoveKey(key);
596  }
597 
598  template<typename ValueFilter>
599  friend size_type erase_if(unordered_multimap& cont, const ValueFilter& valueFilter)
600  {
601  auto pairFilter = [&valueFilter] (const key_type& key, const mapped_type& mapped)
602  { return valueFilter(const_reference(key, mapped)); };
603  return cont.mHashMultiMap.Remove(pairFilter);
604  }
605 
606  //iterator insert(node_type&& node)
607  //iterator insert(const_iterator, node_type&& node)
608  //node_type extract(const_iterator where)
609  //node_type extract(const key_type& key)
610  //void merge(...)
611 
612  //size_type max_bucket_count() const noexcept
613  //size_type bucket_count() const noexcept
614  //size_type bucket_size(size_type bucketIndex) const
615  //local_iterator begin(size_type bucketIndex)
616  //const_local_iterator begin(size_type bucketIndex) const
617  //local_iterator end(size_type bucketIndex)
618  //const_local_iterator end(size_type bucketIndex) const
619  //const_local_iterator cbegin(size_type bucketIndex) const
620  //const_local_iterator cend(size_type bucketIndex) const
621  //size_type bucket(const key_type& key) const
622  //float load_factor() const noexcept
623 
624  friend bool operator==(const unordered_multimap& left, const unordered_multimap& right)
625  {
626  if (left.mHashMultiMap.GetKeyCount() != right.mHashMultiMap.GetKeyCount())
627  return false;
628  if (left.mHashMultiMap.GetCount() != right.mHashMultiMap.GetCount())
629  return false;
630  typedef typename HashMultiMap::ConstKeyIterator ConstKeyIterator;
631  for (typename ConstKeyIterator::Reference ref : left.mHashMultiMap.GetKeyBounds())
632  {
633  if (ref.GetCount() == 0)
634  continue;
635  ConstKeyIterator rightKeyIter = right.mHashMultiMap.Find(ref.key);
636  if (!rightKeyIter)
637  return false;
638  if (ref.GetCount() != rightKeyIter->GetCount())
639  return false;
640  if (!std::is_permutation(ref.GetBegin(), ref.GetEnd(), rightKeyIter->GetBegin()))
641  return false;
642  }
643  return true;
644  }
645 
646  friend bool operator!=(const unordered_multimap& left, const unordered_multimap& right)
647  {
648  return !(left == right);
649  }
650 
651 private:
652  static HashMultiMap pvCreateMultiMap(unordered_multimap&& right, const allocator_type& alloc)
653  {
654  if (right.get_allocator() == alloc)
655  return std::move(right.mHashMultiMap);
656  HashMultiMap hashMultiMap(right.mHashMultiMap.GetHashTraits(), MemManager(alloc));
657  for (reference ref : right)
658  hashMultiMap.Add(ref.first, std::move(ref.second));
659  right.clear();
660  return hashMultiMap;
661  }
662 
663  template<typename Iterator, typename IteratorProxy, typename HashMultiMap, typename KeyIterator>
664  static std::pair<Iterator, Iterator> pvEqualRange(HashMultiMap& hashMultiMap,
665  KeyIterator keyIter)
666  {
667  Iterator end = IteratorProxy(hashMultiMap.GetEnd());
668  if (!keyIter)
669  return { end, end };
670  size_t count = keyIter->GetCount();
671  if (count == 0)
672  return { end, end };
673  Iterator first = IteratorProxy(hashMultiMap.MakeIterator(keyIter, 0));
674  Iterator last = IteratorProxy(hashMultiMap.MakeIterator(keyIter, count));
675  return { first, last };
676  }
677 
678  template<typename... KeyArgs, typename... MappedArgs>
679  iterator pvEmplace(std::tuple<KeyArgs...>&& keyArgs, std::tuple<MappedArgs...>&& mappedArgs)
680  {
681  typedef typename HashMultiMap::KeyValueTraits
682  ::template ValueCreator<MappedArgs...> MappedCreator;
683  return pvInsert(std::move(keyArgs),
684  MappedCreator(mHashMultiMap.GetMemManager(), std::move(mappedArgs)));
685  }
686 
687  template<typename... KeyArgs, typename MappedCreator>
688  iterator pvInsert(std::tuple<KeyArgs...>&& keyArgs, MappedCreator&& mappedCreator)
689  {
690  MemManager& memManager = mHashMultiMap.GetMemManager();
693  typedef typename KeyManager::template Creator<KeyArgs...> KeyCreator;
694  KeyBuffer keyBuffer;
695  KeyCreator(memManager, std::move(keyArgs))(keyBuffer.GetPtr());
696  iterator resIter;
697  try
698  {
699  resIter = pvInsert(std::forward_as_tuple(std::move(keyBuffer.Get())),
700  std::forward<MappedCreator>(mappedCreator));
701  }
702  catch (...)
703  {
704  KeyManager::Destroy(memManager, keyBuffer.Get());
705  throw;
706  }
707  KeyManager::Destroy(memManager, keyBuffer.Get());
708  return resIter;
709  }
710 
711  template<typename RKey, typename MappedCreator,
712  typename Key = typename std::decay<RKey>::type>
714  iterator> pvInsert(std::tuple<RKey>&& key, MappedCreator&& mappedCreator)
715  {
716  return IteratorProxy(mHashMultiMap.AddCrt(
717  std::forward<RKey>(std::get<0>(key)), std::forward<MappedCreator>(mappedCreator)));
718  }
719 
720  template<typename Iterator, typename Sentinel>
722  void> pvInsertRange(Iterator begin, Sentinel end)
723  {
724  mHashMultiMap.Add(std::move(begin), std::move(end));
725  }
726 
727  template<typename Iterator, typename Sentinel>
729  void> pvInsertRange(Iterator begin, Sentinel end)
730  {
731  for (Iterator iter = std::move(begin); iter != end; ++iter)
732  insert(*iter);
733  }
734 
735 private:
736  HashMultiMap mHashMultiMap;
737 };
738 
748 template<typename TKey, typename TMapped,
749  typename THashFunc = HashCoder<TKey>,
750  typename TEqualFunc = std::equal_to<TKey>,
751  typename TAllocator = std::allocator<std::pair<const TKey, TMapped>>>
752 class unordered_multimap_open : public unordered_multimap<TKey, TMapped, THashFunc, TEqualFunc, TAllocator,
753  HashMultiMap<TKey, TMapped, HashTraitsStd<TKey, THashFunc, TEqualFunc, HashBucketOpenDefault>,
754  MemManagerStd<TAllocator>>>
755 {
756 private:
757  typedef unordered_multimap<TKey, TMapped, THashFunc, TEqualFunc, TAllocator,
760 
761 public:
762  using typename UnorderedMultiMap::key_type;
763  using typename UnorderedMultiMap::mapped_type;
764  using typename UnorderedMultiMap::size_type;
765  using typename UnorderedMultiMap::value_type;
766  using typename UnorderedMultiMap::const_reference;
767 
768 public:
769  using UnorderedMultiMap::UnorderedMultiMap;
770 
771  unordered_multimap_open() {} // clang 3.6
772 
773  unordered_multimap_open& operator=(std::initializer_list<value_type> values)
774  {
776  return *this;
777  }
778 
779  friend void swap(unordered_multimap_open& left, unordered_multimap_open& right) noexcept
780  {
781  left.swap(right);
782  }
783 
784  template<typename ValueFilter>
785  friend size_type erase_if(unordered_multimap_open& cont, const ValueFilter& valueFilter)
786  {
787  auto pairFilter = [&valueFilter] (const key_type& key, const mapped_type& mapped)
788  { return valueFilter(const_reference(key, mapped)); };
789  return cont.get_nested_container().Remove(pairFilter);
790  }
791 };
792 
793 #ifdef MOMO_HAS_DEDUCTION_GUIDES
794 
795 #define MOMO_DECLARE_DEDUCTION_GUIDES(unordered_multimap) \
796 template<typename Iterator, \
797  typename Value = typename std::iterator_traits<Iterator>::value_type, \
798  typename Key = std::decay_t<typename Value::first_type>, \
799  typename Mapped = std::decay_t<typename Value::second_type>> \
800 unordered_multimap(Iterator, Iterator) \
801  -> unordered_multimap<Key, Mapped>; \
802 template<typename Iterator, \
803  typename Value = typename std::iterator_traits<Iterator>::value_type, \
804  typename Key = std::decay_t<typename Value::first_type>, \
805  typename Mapped = std::decay_t<typename Value::second_type>, \
806  typename Allocator = std::allocator<std::pair<const Key, Mapped>>, \
807  typename = internal::unordered_checker<Key, Allocator, HashCoder<Key>>> \
808 unordered_multimap(Iterator, Iterator, size_t, Allocator = Allocator()) \
809  -> unordered_multimap<Key, Mapped, HashCoder<Key>, std::equal_to<Key>, Allocator>; \
810 template<typename Iterator, typename HashFunc, \
811  typename Value = typename std::iterator_traits<Iterator>::value_type, \
812  typename Key = std::decay_t<typename Value::first_type>, \
813  typename Mapped = std::decay_t<typename Value::second_type>, \
814  typename Allocator = std::allocator<std::pair<const Key, Mapped>>, \
815  typename = internal::unordered_checker<Key, Allocator, HashFunc>> \
816 unordered_multimap(Iterator, Iterator, size_t, HashFunc, Allocator = Allocator()) \
817  -> unordered_multimap<Key, Mapped, HashFunc, std::equal_to<Key>, Allocator>; \
818 template<typename Iterator, typename HashFunc, typename EqualFunc, \
819  typename Value = typename std::iterator_traits<Iterator>::value_type, \
820  typename Key = std::decay_t<typename Value::first_type>, \
821  typename Mapped = std::decay_t<typename Value::second_type>, \
822  typename Allocator = std::allocator<std::pair<const Key, Mapped>>, \
823  typename = internal::unordered_checker<Key, Allocator, HashFunc, EqualFunc>> \
824 unordered_multimap(Iterator, Iterator, size_t, HashFunc, EqualFunc, Allocator = Allocator()) \
825  -> unordered_multimap<Key, Mapped, HashFunc, EqualFunc, Allocator>; \
826 template<typename CKey, typename Mapped, \
827  typename Key = std::remove_const_t<CKey>> \
828 unordered_multimap(std::initializer_list<std::pair<CKey, Mapped>>) \
829  -> unordered_multimap<Key, Mapped>; \
830 template<typename CKey, typename Mapped, \
831  typename Key = std::remove_const_t<CKey>, \
832  typename Allocator = std::allocator<std::pair<const Key, Mapped>>, \
833  typename = internal::unordered_checker<Key, Allocator, HashCoder<Key>>> \
834 unordered_multimap(std::initializer_list<std::pair<CKey, Mapped>>, size_t, Allocator = Allocator()) \
835  -> unordered_multimap<Key, Mapped, HashCoder<Key>, std::equal_to<Key>, Allocator>; \
836 template<typename CKey, typename Mapped, typename HashFunc, \
837  typename Key = std::remove_const_t<CKey>, \
838  typename Allocator = std::allocator<std::pair<const Key, Mapped>>, \
839  typename = internal::unordered_checker<Key, Allocator, HashFunc>> \
840 unordered_multimap(std::initializer_list<std::pair<CKey, Mapped>>, size_t, HashFunc, Allocator = Allocator()) \
841  -> unordered_multimap<Key, Mapped, HashFunc, std::equal_to<Key>, Allocator>; \
842 template<typename CKey, typename Mapped, typename HashFunc, typename EqualFunc, \
843  typename Key = std::remove_const_t<CKey>, \
844  typename Allocator = std::allocator<std::pair<const Key, Mapped>>, \
845  typename = internal::unordered_checker<Key, Allocator, HashFunc, EqualFunc>> \
846 unordered_multimap(std::initializer_list<std::pair<CKey, Mapped>>, size_t, HashFunc, EqualFunc, Allocator = Allocator()) \
847  -> unordered_multimap<Key, Mapped, HashFunc, EqualFunc, Allocator>;
848 
849 MOMO_DECLARE_DEDUCTION_GUIDES(unordered_multimap)
850 MOMO_DECLARE_DEDUCTION_GUIDES(unordered_multimap_open)
851 
852 #undef MOMO_DECLARE_DEDUCTION_GUIDES
853 
854 #ifdef MOMO_HAS_CONTAINERS_RANGES
855 
856 #define MOMO_DECLARE_DEDUCTION_GUIDES_RANGES(unordered_multimap) \
857 template<std::ranges::input_range Range, \
858  typename Value = std::ranges::range_value_t<Range>, \
859  typename Key = std::decay_t<typename Value::first_type>, \
860  typename Mapped = std::decay_t<typename Value::second_type>> \
861 unordered_multimap(std::from_range_t, Range&&) \
862  -> unordered_multimap<Key, Mapped>; \
863 template<std::ranges::input_range Range, \
864  typename Value = std::ranges::range_value_t<Range>, \
865  typename Key = std::decay_t<typename Value::first_type>, \
866  typename Mapped = std::decay_t<typename Value::second_type>, \
867  typename Allocator = std::allocator<std::pair<const Key, Mapped>>, \
868  typename = internal::unordered_checker<Key, Allocator, HashCoder<Key>>> \
869 unordered_multimap(std::from_range_t, Range&&, size_t, Allocator = Allocator()) \
870  -> unordered_multimap<Key, Mapped, HashCoder<Key>, std::equal_to<Key>, Allocator>; \
871 template<std::ranges::input_range Range, typename HashFunc, \
872  typename Value = std::ranges::range_value_t<Range>, \
873  typename Key = std::decay_t<typename Value::first_type>, \
874  typename Mapped = std::decay_t<typename Value::second_type>, \
875  typename Allocator = std::allocator<std::pair<const Key, Mapped>>, \
876  typename = internal::unordered_checker<Key, Allocator, HashFunc>> \
877 unordered_multimap(std::from_range_t, Range&&, size_t, HashFunc, Allocator = Allocator()) \
878  -> unordered_multimap<Key, Mapped, HashFunc, std::equal_to<Key>, Allocator>; \
879 template<std::ranges::input_range Range, typename HashFunc, typename EqualFunc, \
880  typename Value = std::ranges::range_value_t<Range>, \
881  typename Key = std::decay_t<typename Value::first_type>, \
882  typename Mapped = std::decay_t<typename Value::second_type>, \
883  typename Allocator = std::allocator<std::pair<const Key, Mapped>>, \
884  typename = internal::unordered_checker<Key, Allocator, HashFunc, EqualFunc>> \
885 unordered_multimap(std::from_range_t, Range&&, size_t, HashFunc, EqualFunc, Allocator = Allocator()) \
886  -> unordered_multimap<Key, Mapped, HashFunc, EqualFunc, Allocator>;
887 
888 MOMO_DECLARE_DEDUCTION_GUIDES_RANGES(unordered_multimap)
889 MOMO_DECLARE_DEDUCTION_GUIDES_RANGES(unordered_multimap_open)
890 
891 #undef MOMO_DECLARE_DEDUCTION_GUIDES_RANGES
892 
893 #endif // MOMO_HAS_CONTAINERS_RANGES
894 
895 #endif // MOMO_HAS_DEDUCTION_GUIDES
896 
897 } // namespace stdish
898 
899 } // namespace momo
900 
901 #endif // MOMO_INCLUDE_GUARD_STDISH_UNORDERED_MULTIMAP
momo::stdish::unordered_multimap::swap
friend void swap(unordered_multimap &left, unordered_multimap &right) noexcept
Definition: unordered_multimap.h:308
momo::stdish::unordered_multimap::equal_range
MOMO_FORCEINLINE momo::internal::EnableIf< IsValidKeyArg< KeyArg >::value, std::pair< iterator, iterator > > equal_range(const KeyArg &key)
Definition: unordered_multimap.h:465
momo::internal::ObjectManager
Definition: ObjectManager.h:212
momo::HashMultiMap::Iterator
internal::HashMultiMapIterator< KeyIterator, Settings > Iterator
Definition: HashMultiMap.h:674
momo::stdish::unordered_multimap::max_size
size_type max_size() const noexcept
Definition: unordered_multimap.h:371
momo::stdish::unordered_multimap::insert
void insert(std::initializer_list< value_type > values)
Definition: unordered_multimap.h:490
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap(const unordered_multimap &right)
Definition: unordered_multimap.h:254
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap(unordered_multimap &&right, const momo::internal::Identity< allocator_type > &alloc) noexcept(std::is_empty< allocator_type >::value)
Definition: unordered_multimap.h:247
momo::internal::HashMultiMapIterator
Definition: HashMultiMap.h:182
momo::HashMultiMap::MakeIterator
ConstIterator MakeIterator(ConstKeyIterator keyIter, size_t valueIndex=0) const
Definition: HashMultiMap.h:1136
momo::internal::HashDerivedIterator::Reference
TReference< typename BaseIterator::Reference > Reference
Definition: IteratorUtility.h:381
momo::stdish::unordered_multimap::allocator_type
TAllocator allocator_type
Definition: unordered_multimap.h:76
momo::stdish::unordered_multimap::mapped_type
TMapped mapped_type
Definition: unordered_multimap.h:73
momo::stdish::unordered_multimap::erase
iterator erase(iterator where)
Definition: unordered_multimap.h:564
momo::stdish::unordered_multimap::swap
void swap(unordered_multimap &right) noexcept
Definition: unordered_multimap.h:301
momo::stdish::unordered_multimap_open::unordered_multimap_open
unordered_multimap_open()
Definition: unordered_multimap.h:771
momo::HashMultiMap::HashTraits
THashTraits HashTraits
Definition: HashMultiMap.h:564
momo::stdish::unordered_multimap::cbegin
const_iterator cbegin() const noexcept
Definition: unordered_multimap.h:343
momo::stdish::unordered_multimap::get_nested_container
const nested_container_type & get_nested_container() const noexcept
Definition: unordered_multimap.h:313
momo::stdish::unordered_multimap::end
const_iterator end() const noexcept
Definition: unordered_multimap.h:333
set_map_utility.h
momo::stdish::unordered_multimap::count
MOMO_FORCEINLINE momo::internal::EnableIf< IsValidKeyArg< KeyArg >::value, size_type > count(const KeyArg &key) const
Definition: unordered_multimap.h:426
momo::stdish::unordered_multimap::operator!=
friend bool operator!=(const unordered_multimap &left, const unordered_multimap &right)
Definition: unordered_multimap.h:646
momo::stdish::unordered_multimap::erase
iterator erase(const_iterator where)
Definition: unordered_multimap.h:554
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap()
Definition: unordered_multimap.h:123
momo::stdish::unordered_multimap::begin
const_iterator begin() const noexcept
Definition: unordered_multimap.h:323
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap(Iterator first, Iterator last, size_type bucketCount, const allocator_type &alloc=allocator_type())
Definition: unordered_multimap.h:157
momo::stdish::unordered_multimap::size_type
size_t size_type
Definition: unordered_multimap.h:80
momo::stdish::unordered_multimap::find
MOMO_FORCEINLINE momo::internal::EnableIf< IsValidKeyArg< KeyArg >::value, iterator > find(const KeyArg &key)
Definition: unordered_multimap.h:413
momo::stdish::unordered_multimap::operator=
unordered_multimap & operator=(std::initializer_list< value_type > values)
Definition: unordered_multimap.h:293
momo::stdish::unordered_multimap
momo::stdish::unordered_multimap is similar to std::unordered_multimap, but much more efficient in me...
Definition: unordered_multimap.h:65
momo::stdish::unordered_multimap::nested_container_type
HashMultiMap nested_container_type
Definition: unordered_multimap.h:78
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap(Iterator first, Iterator last, size_type bucketCount, const hasher &hashFunc, const key_equal &equalFunc, const allocator_type &alloc=allocator_type())
Definition: unordered_multimap.h:173
momo::stdish::unordered_multimap::emplace
iterator emplace(KeyArg &&keyArg, MappedArg &&mappedArg)
Definition: unordered_multimap.h:528
momo::stdish::unordered_multimap::emplace
iterator emplace(std::piecewise_construct_t, std::tuple< KeyArgs... > keyArgs, std::tuple< MappedArgs... > mappedArgs)
Definition: unordered_multimap.h:541
momo::internal::HashMultiMapIterator::GetKeyIterator
KeyIterator GetKeyIterator() const noexcept
Definition: HashMultiMap.h:249
momo::stdish::unordered_multimap::get_nested_container
nested_container_type & get_nested_container() noexcept
Definition: unordered_multimap.h:318
momo::stdish::unordered_multimap::begin
iterator begin() noexcept
Definition: unordered_multimap.h:328
momo::MemManagerStd
MemManagerStd uses allocator<unsigned char>::allocate and deallocate
Definition: MemManager.h:177
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap(unordered_multimap &&right) noexcept
Definition: unordered_multimap.h:242
momo::stdish::unordered_multimap::key_equal
TEqualFunc key_equal
Definition: unordered_multimap.h:75
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap(Iterator first, Iterator last, size_type bucketCount, const hasher &hashFunc, const allocator_type &alloc=allocator_type())
Definition: unordered_multimap.h:165
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap(size_type bucketCount, const allocator_type &alloc=allocator_type())
Definition: unordered_multimap.h:132
momo::stdish::unordered_multimap::emplace
iterator emplace()
Definition: unordered_multimap.h:504
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap(Iterator first, Iterator last)
Definition: unordered_multimap.h:151
momo::stdish::unordered_multimap::emplace_hint
iterator emplace_hint(const_iterator, ValueArg &&valueArg)
Definition: unordered_multimap.h:522
momo::stdish::unordered_multimap::cend
const_iterator cend() const noexcept
Definition: unordered_multimap.h:348
momo::stdish::unordered_multimap::end
iterator end() noexcept
Definition: unordered_multimap.h:338
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap(size_type bucketCount, const hasher &hashFunc, const key_equal &equalFunc, const allocator_type &alloc=allocator_type())
Definition: unordered_multimap.h:144
momo::stdish::unordered_multimap::erase_if
friend size_type erase_if(unordered_multimap &cont, const ValueFilter &valueFilter)
Definition: unordered_multimap.h:599
momo::stdish::unordered_multimap_open::mapped_type
TMapped mapped_type
Definition: unordered_multimap.h:73
momo::stdish::unordered_multimap_open::key_type
TKey key_type
Definition: unordered_multimap.h:72
momo::stdish::unordered_multimap::operator=
unordered_multimap & operator=(unordered_multimap &&right) noexcept(std::is_empty< allocator_type >::value||std::allocator_traits< allocator_type >::propagate_on_container_move_assignment::value)
Definition: unordered_multimap.h:267
momo::stdish::unordered_multimap::operator=
unordered_multimap & operator=(const unordered_multimap &right)
Definition: unordered_multimap.h:281
momo::stdish::unordered_multimap::equal_range
MOMO_FORCEINLINE std::pair< iterator, iterator > equal_range(const key_type &key)
Definition: unordered_multimap.h:450
momo::stdish::unordered_multimap::erase
size_type erase(const key_type &key)
Definition: unordered_multimap.h:593
momo::internal::EnableIf
typename std::enable_if< value, Type >::type EnableIf
Definition: Utility.h:198
momo::stdish::unordered_multimap::emplace
iterator emplace(ValueArg &&valueArg)
Definition: unordered_multimap.h:515
momo::stdish::unordered_multimap::iterator
momo::internal::HashDerivedIterator< typename HashMultiMap::Iterator, momo::internal::MapReferenceStd > iterator
Definition: unordered_multimap.h:86
momo::stdish::unordered_multimap_open
momo::stdish::unordered_multimap_open is similar to std::unordered_multimap, but much more efficient ...
Definition: unordered_multimap.h:755
momo::stdish::unordered_multimap::get_allocator
allocator_type get_allocator() const noexcept
Definition: unordered_multimap.h:366
momo
Definition: Array.h:26
momo::stdish::unordered_multimap_open::swap
friend void swap(unordered_multimap_open &left, unordered_multimap_open &right) noexcept
Definition: unordered_multimap.h:779
momo::stdish::unordered_multimap::find
MOMO_FORCEINLINE const_iterator find(const key_type &key) const
Definition: unordered_multimap.h:394
momo::internal::IteratorPointer
Definition: IteratorUtility.h:264
momo::stdish::unordered_multimap::hasher
THashFunc hasher
Definition: unordered_multimap.h:74
momo::stdish::unordered_multimap::erase
iterator erase(const_iterator first, const_iterator last)
Definition: unordered_multimap.h:569
momo::internal::Identity
EnableIf< true, Type > Identity
Definition: Utility.h:201
momo::stdish::unordered_multimap::const_pointer
const_iterator::Pointer const_pointer
Definition: unordered_multimap.h:93
momo::internal::HashDerivedIterator::BaseIterator
TBaseIterator BaseIterator
Definition: IteratorUtility.h:378
momo::stdish::unordered_multimap_open::erase_if
friend size_type erase_if(unordered_multimap_open &cont, const ValueFilter &valueFilter)
Definition: unordered_multimap.h:785
momo::stdish::unordered_multimap_open::size_type
size_t size_type
Definition: unordered_multimap.h:80
momo::stdish::unordered_multimap::insert
momo::internal::EnableIf< std::is_constructible< value_type, ValueArg && >::value, iterator > insert(const_iterator, ValueArg &&valueArg)
Definition: unordered_multimap.h:479
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap(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_multimap.h:198
momo::stdish::unordered_multimap::find
MOMO_FORCEINLINE momo::internal::EnableIf< IsValidKeyArg< KeyArg >::value, const_iterator > find(const KeyArg &key) const
Definition: unordered_multimap.h:406
MOMO_DECLARE_PROXY_CONSTRUCTOR
#define MOMO_DECLARE_PROXY_CONSTRUCTOR(Object)
Definition: Utility.h:107
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap(std::initializer_list< momo::internal::Identity< value_type >> values, size_type bucketCount, const hasher &hashFunc, const allocator_type &alloc=allocator_type())
Definition: unordered_multimap.h:192
momo::stdish::unordered_multimap::emplace_hint
iterator emplace_hint(const_iterator, KeyArg &&keyArg, MappedArg &&mappedArg)
Definition: unordered_multimap.h:535
momo::stdish::unordered_multimap::empty
MOMO_NODISCARD bool empty() const noexcept
Definition: unordered_multimap.h:381
momo::stdish::unordered_multimap::insert
momo::internal::EnableIf< std::is_constructible< value_type, ValueArg && >::value, iterator > insert(ValueArg &&valueArg)
Definition: unordered_multimap.h:472
momo::stdish::unordered_multimap::key_eq
key_equal key_eq() const
Definition: unordered_multimap.h:361
momo::internal::HashDerivedIterator
Definition: IteratorUtility.h:376
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap(const unordered_multimap &right, const momo::internal::Identity< allocator_type > &alloc)
Definition: unordered_multimap.h:259
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap(std::initializer_list< momo::internal::Identity< value_type >> values, size_type bucketCount, const allocator_type &alloc=allocator_type())
Definition: unordered_multimap.h:186
momo::stdish::unordered_multimap::difference_type
ptrdiff_t difference_type
Definition: unordered_multimap.h:81
momo::internal::ObjectBuffer
Definition: ObjectManager.h:161
momo::stdish::unordered_multimap::operator==
friend bool operator==(const unordered_multimap &left, const unordered_multimap &right)
Definition: unordered_multimap.h:624
std
Definition: ArrayUtility.h:308
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap(std::initializer_list< momo::internal::Identity< value_type >> values)
Definition: unordered_multimap.h:181
MOMO_FORCEINLINE
#define MOMO_FORCEINLINE
Definition: UserSettings.h:114
momo::stdish::unordered_multimap::reference
iterator::Reference reference
Definition: unordered_multimap.h:89
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap(size_type bucketCount, const hasher &hashFunc, const allocator_type &alloc=allocator_type())
Definition: unordered_multimap.h:138
momo::stdish::unordered_multimap::clear
void clear() noexcept
Definition: unordered_multimap.h:386
momo::stdish::unordered_multimap::find
MOMO_FORCEINLINE iterator find(const key_type &key)
Definition: unordered_multimap.h:399
momo::HashMultiMap::GetEnd
ConstIterator GetEnd() const noexcept
Definition: HashMultiMap.h:828
momo::stdish::unordered_multimap::hash_function
hasher hash_function() const
Definition: unordered_multimap.h:356
momo::HashMultiMap
Definition: HashMultiMap.h:560
momo::HashMultiMap::MemManager
TMemManager MemManager
Definition: HashMultiMap.h:565
momo::stdish::unordered_multimap::equal_range
MOMO_FORCEINLINE std::pair< const_iterator, const_iterator > equal_range(const key_type &key) const
Definition: unordered_multimap.h:444
momo::stdish::unordered_multimap::unordered_multimap
unordered_multimap(const allocator_type &alloc)
Definition: unordered_multimap.h:127
momo::stdish::unordered_multimap::equal_range
MOMO_FORCEINLINE momo::internal::EnableIf< IsValidKeyArg< KeyArg >::value, std::pair< const_iterator, const_iterator > > equal_range(const KeyArg &key) const
Definition: unordered_multimap.h:457
momo::stdish::unordered_multimap::const_iterator
iterator::ConstIterator const_iterator
Definition: unordered_multimap.h:87
momo::stdish::unordered_multimap::key_type
TKey key_type
Definition: unordered_multimap.h:72
momo::stdish::unordered_multimap::size
size_type size() const noexcept
Definition: unordered_multimap.h:376
momo::stdish::unordered_multimap::count
MOMO_FORCEINLINE size_type count(const key_type &key) const
Definition: unordered_multimap.h:418
momo::stdish::unordered_multimap_open::const_reference
const_iterator::Reference const_reference
Definition: unordered_multimap.h:90
momo::stdish::unordered_multimap::const_reference
const_iterator::Reference const_reference
Definition: unordered_multimap.h:90
momo::stdish::unordered_multimap::value_type
std::pair< const key_type, mapped_type > value_type
Definition: unordered_multimap.h:83
momo::stdish::unordered_multimap::contains
MOMO_FORCEINLINE momo::internal::EnableIf< IsValidKeyArg< KeyArg >::value, bool > contains(const KeyArg &key) const
Definition: unordered_multimap.h:439
momo::stdish::unordered_multimap::pointer
iterator::Pointer pointer
Definition: unordered_multimap.h:92
momo::stdish::unordered_multimap::insert
void insert(Iterator first, Iterator last)
Definition: unordered_multimap.h:485
momo::stdish::unordered_multimap_open::operator=
unordered_multimap_open & operator=(std::initializer_list< value_type > values)
Definition: unordered_multimap.h:773
momo::stdish::unordered_multimap::contains
MOMO_FORCEINLINE bool contains(const key_type &key) const
Definition: unordered_multimap.h:432
MOMO_ASSERT
#define MOMO_ASSERT(expr)
Definition: UserSettings.h:162
momo::stdish::unordered_multimap::~unordered_multimap
~unordered_multimap()=default
momo::stdish::unordered_multimap::emplace_hint
iterator emplace_hint(const_iterator)
Definition: unordered_multimap.h:509
MOMO_DECLARE_PROXY_FUNCTION
#define MOMO_DECLARE_PROXY_FUNCTION(Object, Func, Result)
Definition: Utility.h:116
MOMO_NODISCARD
#define MOMO_NODISCARD
Definition: UserSettings.h:203
momo::stdish::unordered_multimap::emplace_hint
iterator emplace_hint(const_iterator, std::piecewise_construct_t, std::tuple< KeyArgs... > keyArgs, std::tuple< MappedArgs... > mappedArgs)
Definition: unordered_multimap.h:548
momo::internal::MapReferenceStd
Definition: MapUtility.h:68