heap_merge.hpp 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. // boost heap: heap merge algorithms
  2. //
  3. // Copyright (C) 2011 Tim Blechmann
  4. //
  5. // Distributed under the Boost Software License, Version 1.0. (See
  6. // accompanying file LICENSE_1_0.txt or copy at
  7. // http://www.boost.org/LICENSE_1_0.txt)
  8. #ifndef BOOST_HEAP_MERGE_HPP
  9. #define BOOST_HEAP_MERGE_HPP
  10. #include <boost/concept/assert.hpp>
  11. #include <boost/heap/heap_concepts.hpp>
  12. #include <boost/type_traits/is_same.hpp>
  13. #ifdef BOOST_HAS_PRAGMA_ONCE
  14. #pragma once
  15. #endif
  16. namespace boost {
  17. namespace heap {
  18. namespace detail {
  19. template <typename Heap1, typename Heap2>
  20. struct heap_merge_emulate
  21. {
  22. struct dummy_reserver
  23. {
  24. static void reserve (Heap1 & lhs, std::size_t required_size)
  25. {}
  26. };
  27. struct reserver
  28. {
  29. static void reserve (Heap1 & lhs, std::size_t required_size)
  30. {
  31. lhs.reserve(required_size);
  32. }
  33. };
  34. typedef typename boost::mpl::if_c<Heap1::has_reserve,
  35. reserver,
  36. dummy_reserver>::type space_reserver;
  37. static void merge(Heap1 & lhs, Heap2 & rhs)
  38. {
  39. if (Heap1::constant_time_size && Heap2::constant_time_size) {
  40. if (Heap1::has_reserve) {
  41. std::size_t required_size = lhs.size() + rhs.size();
  42. space_reserver::reserve(lhs, required_size);
  43. }
  44. }
  45. // FIXME: container adaptors could benefit from first appending all elements and then restoring the heap order
  46. // FIXME: optimize: if we have ordered iterators and we can efficiently insert keys with a below the lowest key in the heap
  47. // d-ary, b and fibonacci heaps fall into this category
  48. while (!rhs.empty()) {
  49. lhs.push(rhs.top());
  50. rhs.pop();
  51. }
  52. lhs.set_stability_count((std::max)(lhs.get_stability_count(),
  53. rhs.get_stability_count()));
  54. rhs.set_stability_count(0);
  55. }
  56. };
  57. template <typename Heap>
  58. struct heap_merge_same_mergable
  59. {
  60. static void merge(Heap & lhs, Heap & rhs)
  61. {
  62. lhs.merge(rhs);
  63. }
  64. };
  65. template <typename Heap>
  66. struct heap_merge_same
  67. {
  68. static const bool is_mergable = Heap::is_mergable;
  69. typedef typename boost::mpl::if_c<is_mergable,
  70. heap_merge_same_mergable<Heap>,
  71. heap_merge_emulate<Heap, Heap>
  72. >::type heap_merger;
  73. static void merge(Heap & lhs, Heap & rhs)
  74. {
  75. heap_merger::merge(lhs, rhs);
  76. }
  77. };
  78. } /* namespace detail */
  79. /** merge rhs into lhs
  80. *
  81. * \b Effect: lhs contains all elements that have been part of rhs, rhs is empty.
  82. *
  83. * */
  84. template <typename Heap1,
  85. typename Heap2
  86. >
  87. void heap_merge(Heap1 & lhs, Heap2 & rhs)
  88. {
  89. BOOST_CONCEPT_ASSERT((boost::heap::PriorityQueue<Heap1>));
  90. BOOST_CONCEPT_ASSERT((boost::heap::PriorityQueue<Heap2>));
  91. // if this assertion is triggered, the value_compare types are incompatible
  92. BOOST_STATIC_ASSERT((boost::is_same<typename Heap1::value_compare, typename Heap2::value_compare>::value));
  93. const bool same_heaps = boost::is_same<Heap1, Heap2>::value;
  94. typedef typename boost::mpl::if_c<same_heaps,
  95. detail::heap_merge_same<Heap1>,
  96. detail::heap_merge_emulate<Heap1, Heap2>
  97. >::type heap_merger;
  98. heap_merger::merge(lhs, rhs);
  99. }
  100. } /* namespace heap */
  101. } /* namespace boost */
  102. #endif /* BOOST_HEAP_MERGE_HPP */