relaxed_heap.hpp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647
  1. // Copyright 2004 The Trustees of Indiana University.
  2. // Use, modification and distribution is subject to the Boost Software
  3. // License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  4. // http://www.boost.org/LICENSE_1_0.txt)
  5. // Authors: Douglas Gregor
  6. // Andrew Lumsdaine
  7. #ifndef BOOST_RELAXED_HEAP_HEADER
  8. #define BOOST_RELAXED_HEAP_HEADER
  9. #include <functional>
  10. #include <boost/property_map/property_map.hpp>
  11. #include <boost/optional.hpp>
  12. #include <vector>
  13. #include <climits> // for CHAR_BIT
  14. #include <boost/none.hpp>
  15. #ifdef BOOST_RELAXED_HEAP_DEBUG
  16. # include <iostream>
  17. #endif // BOOST_RELAXED_HEAP_DEBUG
  18. #if defined(BOOST_MSVC)
  19. # pragma warning(push)
  20. # pragma warning(disable:4355) // complaint about using 'this' to
  21. #endif // initialize a member
  22. namespace boost {
  23. template<typename IndexedType,
  24. typename Compare = std::less<IndexedType>,
  25. typename ID = identity_property_map>
  26. class relaxed_heap
  27. {
  28. struct group;
  29. typedef relaxed_heap self_type;
  30. typedef std::size_t rank_type;
  31. public:
  32. typedef IndexedType value_type;
  33. typedef rank_type size_type;
  34. private:
  35. /**
  36. * The kind of key that a group has. The actual values are discussed
  37. * in-depth in the documentation of the @c kind field of the @c group
  38. * structure. Note that the order of the enumerators *IS* important
  39. * and must not be changed.
  40. */
  41. enum group_key_kind { smallest_key, stored_key, largest_key };
  42. struct group {
  43. explicit group(group_key_kind kind = largest_key)
  44. : kind(kind), parent(this), rank(0) { }
  45. /** The value associated with this group. This value is only valid
  46. * when @c kind!=largest_key (which indicates a deleted
  47. * element). Note that the use of boost::optional increases the
  48. * memory requirements slightly but does not result in extraneous
  49. * memory allocations or deallocations. The optional could be
  50. * eliminated when @c value_type is a model of
  51. * DefaultConstructible.
  52. */
  53. ::boost::optional<value_type> value;
  54. /**
  55. * The kind of key stored at this group. This may be @c
  56. * smallest_key, which indicates that the key is infinitely small;
  57. * @c largest_key, which indicates that the key is infinitely
  58. * large; or @c stored_key, which means that the key is unknown,
  59. * but its relationship to other keys can be determined via the
  60. * comparison function object.
  61. */
  62. group_key_kind kind;
  63. /// The parent of this group. Will only be NULL for the dummy root group
  64. group* parent;
  65. /// The rank of this group. Equivalent to the number of children in
  66. /// the group.
  67. rank_type rank;
  68. /** The children of this group. For the dummy root group, these are
  69. * the roots. This is an array of length log n containing pointers
  70. * to the child groups.
  71. */
  72. group** children;
  73. };
  74. size_type log_base_2(size_type n) // log2 is a macro on some platforms
  75. {
  76. size_type leading_zeroes = 0;
  77. do {
  78. size_type next = n << 1;
  79. if (n == (next >> 1)) {
  80. ++leading_zeroes;
  81. n = next;
  82. } else {
  83. break;
  84. }
  85. } while (true);
  86. return sizeof(size_type) * CHAR_BIT - leading_zeroes - 1;
  87. }
  88. public:
  89. relaxed_heap(size_type n, const Compare& compare = Compare(),
  90. const ID& id = ID())
  91. : compare(compare), id(id), root(smallest_key), groups(n),
  92. smallest_value(0)
  93. {
  94. if (n == 0) {
  95. root.children = new group*[1];
  96. return;
  97. }
  98. log_n = log_base_2(n);
  99. if (log_n == 0) log_n = 1;
  100. size_type g = n / log_n;
  101. if (n % log_n > 0) ++g;
  102. size_type log_g = log_base_2(g);
  103. size_type r = log_g;
  104. // Reserve an appropriate amount of space for data structures, so
  105. // that we do not need to expand them.
  106. index_to_group.resize(g);
  107. A.resize(r + 1, 0);
  108. root.rank = r + 1;
  109. root.children = new group*[(log_g + 1) * (g + 1)];
  110. for (rank_type i = 0; i < r+1; ++i) root.children[i] = 0;
  111. // Build initial heap
  112. size_type idx = 0;
  113. while (idx < g) {
  114. root.children[r] = &index_to_group[idx];
  115. idx = build_tree(root, idx, r, log_g + 1);
  116. if (idx != g)
  117. r = static_cast<size_type>(log_base_2(g-idx));
  118. }
  119. }
  120. ~relaxed_heap() { delete [] root.children; }
  121. void push(const value_type& x)
  122. {
  123. groups[get(id, x)] = x;
  124. update(x);
  125. }
  126. void update(const value_type& x)
  127. {
  128. group* a = &index_to_group[get(id, x) / log_n];
  129. if (!a->value
  130. || *a->value == x
  131. || compare(x, *a->value)) {
  132. if (a != smallest_value) smallest_value = 0;
  133. a->kind = stored_key;
  134. a->value = x;
  135. promote(a);
  136. }
  137. }
  138. void remove(const value_type& x)
  139. {
  140. group* a = &index_to_group[get(id, x) / log_n];
  141. assert(groups[get(id, x)]);
  142. a->value = x;
  143. a->kind = smallest_key;
  144. promote(a);
  145. smallest_value = a;
  146. pop();
  147. }
  148. value_type& top()
  149. {
  150. find_smallest();
  151. assert(smallest_value->value != none);
  152. return *smallest_value->value;
  153. }
  154. const value_type& top() const
  155. {
  156. find_smallest();
  157. assert(smallest_value->value != none);
  158. return *smallest_value->value;
  159. }
  160. bool empty() const
  161. {
  162. find_smallest();
  163. return !smallest_value || (smallest_value->kind == largest_key);
  164. }
  165. bool contains(const value_type& x) const {
  166. return static_cast<bool>(groups[get(id, x)]);
  167. }
  168. void pop()
  169. {
  170. // Fill in smallest_value. This is the group x.
  171. find_smallest();
  172. group* x = smallest_value;
  173. smallest_value = 0;
  174. // Make x a leaf, giving it the smallest value within its group
  175. rank_type r = x->rank;
  176. group* p = x->parent;
  177. {
  178. assert(x->value != none);
  179. // Find x's group
  180. size_type start = get(id, *x->value) - get(id, *x->value) % log_n;
  181. size_type end = start + log_n;
  182. if (end > groups.size()) end = groups.size();
  183. // Remove the smallest value from the group, and find the new
  184. // smallest value.
  185. groups[get(id, *x->value)].reset();
  186. x->value.reset();
  187. x->kind = largest_key;
  188. for (size_type i = start; i < end; ++i) {
  189. if (groups[i] && (!x->value || compare(*groups[i], *x->value))) {
  190. x->kind = stored_key;
  191. x->value = groups[i];
  192. }
  193. }
  194. }
  195. x->rank = 0;
  196. // Combine prior children of x with x
  197. group* y = x;
  198. for (size_type c = 0; c < r; ++c) {
  199. group* child = x->children[c];
  200. if (A[c] == child) A[c] = 0;
  201. y = combine(y, child);
  202. }
  203. // If we got back something other than x, let y take x's place
  204. if (y != x) {
  205. y->parent = p;
  206. p->children[r] = y;
  207. assert(r == y->rank);
  208. if (A[y->rank] == x)
  209. A[y->rank] = do_compare(y, p)? y : 0;
  210. }
  211. }
  212. #ifdef BOOST_RELAXED_HEAP_DEBUG
  213. /*************************************************************************
  214. * Debugging support *
  215. *************************************************************************/
  216. void dump_tree() { dump_tree(std::cout); }
  217. void dump_tree(std::ostream& out) { dump_tree(out, &root); }
  218. void dump_tree(std::ostream& out, group* p, bool in_progress = false)
  219. {
  220. if (!in_progress) {
  221. out << "digraph heap {\n"
  222. << " edge[dir=\"back\"];\n";
  223. }
  224. size_type p_index = 0;
  225. if (p != &root) while (&index_to_group[p_index] != p) ++p_index;
  226. for (size_type i = 0; i < p->rank; ++i) {
  227. group* c = p->children[i];
  228. if (c) {
  229. size_type c_index = 0;
  230. if (c != &root) while (&index_to_group[c_index] != c) ++c_index;
  231. out << " ";
  232. if (p == &root) out << 'p'; else out << p_index;
  233. out << " -> ";
  234. if (c == &root) out << 'p'; else out << c_index;
  235. if (A[c->rank] == c) out << " [style=\"dotted\"]";
  236. out << ";\n";
  237. dump_tree(out, c, true);
  238. // Emit node information
  239. out << " ";
  240. if (c == &root) out << 'p'; else out << c_index;
  241. out << " [label=\"";
  242. if (c == &root) out << 'p'; else out << c_index;
  243. out << ":";
  244. size_type start = c_index * log_n;
  245. size_type end = start + log_n;
  246. if (end > groups.size()) end = groups.size();
  247. while (start != end) {
  248. if (groups[start]) {
  249. out << " " << get(id, *groups[start]);
  250. if (*groups[start] == *c->value) out << "(*)";
  251. }
  252. ++start;
  253. }
  254. out << '"';
  255. if (do_compare(c, p)) {
  256. out << " ";
  257. if (c == &root) out << 'p'; else out << c_index;
  258. out << ", style=\"filled\", fillcolor=\"gray\"";
  259. }
  260. out << "];\n";
  261. } else {
  262. assert(p->parent == p);
  263. }
  264. }
  265. if (!in_progress) out << "}\n";
  266. }
  267. bool valid()
  268. {
  269. // Check that the ranks in the A array match the ranks of the
  270. // groups stored there. Also, the active groups must be the last
  271. // child of their parent.
  272. for (size_type r = 0; r < A.size(); ++r) {
  273. if (A[r] && A[r]->rank != r) return false;
  274. if (A[r] && A[r]->parent->children[A[r]->parent->rank-1] != A[r])
  275. return false;
  276. }
  277. // The root must have no value and a key of -Infinity
  278. if (root.kind != smallest_key) return false;
  279. return valid(&root);
  280. }
  281. bool valid(group* p)
  282. {
  283. for (size_type i = 0; i < p->rank; ++i) {
  284. group* c = p->children[i];
  285. if (c) {
  286. // Check link structure
  287. if (c->parent != p) return false;
  288. if (c->rank != i) return false;
  289. // A bad group must be active
  290. if (do_compare(c, p) && A[i] != c) return false;
  291. // Check recursively
  292. if (!valid(c)) return false;
  293. } else {
  294. // Only the root may
  295. if (p != &root) return false;
  296. }
  297. }
  298. return true;
  299. }
  300. #endif // BOOST_RELAXED_HEAP_DEBUG
  301. private:
  302. size_type
  303. build_tree(group& parent, size_type idx, size_type r, size_type max_rank)
  304. {
  305. group& this_group = index_to_group[idx];
  306. this_group.parent = &parent;
  307. ++idx;
  308. this_group.children = root.children + (idx * max_rank);
  309. this_group.rank = r;
  310. for (size_type i = 0; i < r; ++i) {
  311. this_group.children[i] = &index_to_group[idx];
  312. idx = build_tree(this_group, idx, i, max_rank);
  313. }
  314. return idx;
  315. }
  316. void find_smallest() const
  317. {
  318. group** roots = root.children;
  319. if (!smallest_value) {
  320. std::size_t i;
  321. for (i = 0; i < root.rank; ++i) {
  322. if (roots[i] &&
  323. (!smallest_value || do_compare(roots[i], smallest_value))) {
  324. smallest_value = roots[i];
  325. }
  326. }
  327. for (i = 0; i < A.size(); ++i) {
  328. if (A[i] && (!smallest_value || do_compare(A[i], smallest_value)))
  329. smallest_value = A[i];
  330. }
  331. }
  332. }
  333. bool do_compare(group* x, group* y) const
  334. {
  335. return (x->kind < y->kind
  336. || (x->kind == y->kind
  337. && x->kind == stored_key
  338. && compare(*x->value, *y->value)));
  339. }
  340. void promote(group* a)
  341. {
  342. assert(a != 0);
  343. rank_type r = a->rank;
  344. group* p = a->parent;
  345. assert(p != 0);
  346. if (do_compare(a, p)) {
  347. // s is the rank + 1 sibling
  348. group* s = p->rank > r + 1? p->children[r + 1] : 0;
  349. // If a is the last child of p
  350. if (r == p->rank - 1) {
  351. if (!A[r]) A[r] = a;
  352. else if (A[r] != a) pair_transform(a);
  353. } else {
  354. assert(s != 0);
  355. if (A[r + 1] == s) active_sibling_transform(a, s);
  356. else good_sibling_transform(a, s);
  357. }
  358. }
  359. }
  360. group* combine(group* a1, group* a2)
  361. {
  362. assert(a1->rank == a2->rank);
  363. if (do_compare(a2, a1)) do_swap(a1, a2);
  364. a1->children[a1->rank++] = a2;
  365. a2->parent = a1;
  366. clean(a1);
  367. return a1;
  368. }
  369. void clean(group* q)
  370. {
  371. if (2 > q->rank) return;
  372. group* qp = q->children[q->rank-1];
  373. rank_type s = q->rank - 2;
  374. group* x = q->children[s];
  375. group* xp = qp->children[s];
  376. assert(s == x->rank);
  377. // If x is active, swap x and xp
  378. if (A[s] == x) {
  379. q->children[s] = xp;
  380. xp->parent = q;
  381. qp->children[s] = x;
  382. x->parent = qp;
  383. }
  384. }
  385. void pair_transform(group* a)
  386. {
  387. #if defined(BOOST_RELAXED_HEAP_DEBUG) && BOOST_RELAXED_HEAP_DEBUG > 1
  388. std::cerr << "- pair transform\n";
  389. #endif
  390. rank_type r = a->rank;
  391. // p is a's parent
  392. group* p = a->parent;
  393. assert(p != 0);
  394. // g is p's parent (a's grandparent)
  395. group* g = p->parent;
  396. assert(g != 0);
  397. // a' <- A(r)
  398. assert(A[r] != 0);
  399. group* ap = A[r];
  400. assert(ap != 0);
  401. // A(r) <- nil
  402. A[r] = 0;
  403. // let a' have parent p'
  404. group* pp = ap->parent;
  405. assert(pp != 0);
  406. // let a' have grandparent g'
  407. group* gp = pp->parent;
  408. assert(gp != 0);
  409. // Remove a and a' from their parents
  410. assert(ap == pp->children[pp->rank-1]); // Guaranteed because ap is active
  411. --pp->rank;
  412. // Guaranteed by caller
  413. assert(a == p->children[p->rank-1]);
  414. --p->rank;
  415. // Note: a, ap, p, pp all have rank r
  416. if (do_compare(pp, p)) {
  417. do_swap(a, ap);
  418. do_swap(p, pp);
  419. do_swap(g, gp);
  420. }
  421. // Assuming k(p) <= k(p')
  422. // make p' the rank r child of p
  423. assert(r == p->rank);
  424. p->children[p->rank++] = pp;
  425. pp->parent = p;
  426. // Combine a, ap into a rank r+1 group c
  427. group* c = combine(a, ap);
  428. // make c the rank r+1 child of g'
  429. assert(gp->rank > r+1);
  430. gp->children[r+1] = c;
  431. c->parent = gp;
  432. #if defined(BOOST_RELAXED_HEAP_DEBUG) && BOOST_RELAXED_HEAP_DEBUG > 1
  433. std::cerr << "After pair transform...\n";
  434. dump_tree();
  435. #endif
  436. if (A[r+1] == pp) A[r+1] = c;
  437. else promote(c);
  438. }
  439. void active_sibling_transform(group* a, group* s)
  440. {
  441. #if defined(BOOST_RELAXED_HEAP_DEBUG) && BOOST_RELAXED_HEAP_DEBUG > 1
  442. std::cerr << "- active sibling transform\n";
  443. #endif
  444. group* p = a->parent;
  445. group* g = p->parent;
  446. // remove a, s from their parents
  447. assert(s->parent == p);
  448. assert(p->children[p->rank-1] == s);
  449. --p->rank;
  450. assert(p->children[p->rank-1] == a);
  451. --p->rank;
  452. rank_type r = a->rank;
  453. A[r+1] = 0;
  454. a = combine(p, a);
  455. group* c = combine(a, s);
  456. // make c the rank r+2 child of g
  457. assert(g->children[r+2] == p);
  458. g->children[r+2] = c;
  459. c->parent = g;
  460. if (A[r+2] == p) A[r+2] = c;
  461. else promote(c);
  462. }
  463. void good_sibling_transform(group* a, group* s)
  464. {
  465. #if defined(BOOST_RELAXED_HEAP_DEBUG) && BOOST_RELAXED_HEAP_DEBUG > 1
  466. std::cerr << "- good sibling transform\n";
  467. #endif
  468. rank_type r = a->rank;
  469. group* c = s->children[s->rank-1];
  470. assert(c->rank == r);
  471. if (A[r] == c) {
  472. #if defined(BOOST_RELAXED_HEAP_DEBUG) && BOOST_RELAXED_HEAP_DEBUG > 1
  473. std::cerr << "- good sibling pair transform\n";
  474. #endif
  475. A[r] = 0;
  476. group* p = a->parent;
  477. // Remove c from its parent
  478. --s->rank;
  479. // Make s the rank r child of p
  480. s->parent = p;
  481. p->children[r] = s;
  482. // combine a, c and let the result by the rank r+1 child of p
  483. assert(p->rank > r+1);
  484. group* x = combine(a, c);
  485. x->parent = p;
  486. p->children[r+1] = x;
  487. if (A[r+1] == s) A[r+1] = x;
  488. else promote(x);
  489. #if defined(BOOST_RELAXED_HEAP_DEBUG) && BOOST_RELAXED_HEAP_DEBUG > 1
  490. dump_tree(std::cerr);
  491. #endif
  492. // pair_transform(a);
  493. } else {
  494. // Clean operation
  495. group* p = a->parent;
  496. s->children[r] = a;
  497. a->parent = s;
  498. p->children[r] = c;
  499. c->parent = p;
  500. promote(a);
  501. }
  502. }
  503. static void do_swap(group*& x, group*& y)
  504. {
  505. group* tmp = x;
  506. x = y;
  507. y = tmp;
  508. }
  509. /// Function object that compares two values in the heap
  510. Compare compare;
  511. /// Mapping from values to indices in the range [0, n).
  512. ID id;
  513. /** The root group of the queue. This group is special because it will
  514. * never store a value, but it acts as a parent to all of the
  515. * roots. Thus, its list of children is the list of roots.
  516. */
  517. group root;
  518. /** Mapping from the group index of a value to the group associated
  519. * with that value. If a value is not in the queue, then the "value"
  520. * field will be empty.
  521. */
  522. std::vector<group> index_to_group;
  523. /** Flat data structure containing the values in each of the
  524. * groups. It will be indexed via the id of the values. The groups
  525. * are each log_n long, with the last group potentially being
  526. * smaller.
  527. */
  528. std::vector< ::boost::optional<value_type> > groups;
  529. /** The list of active groups, indexed by rank. When A[r] is null,
  530. * there is no active group of rank r. Otherwise, A[r] is the active
  531. * group of rank r.
  532. */
  533. std::vector<group*> A;
  534. /** The group containing the smallest value in the queue, which must
  535. * be either a root or an active group. If this group is null, then we
  536. * will need to search for this group when it is needed.
  537. */
  538. mutable group* smallest_value;
  539. /// Cached value log_base_2(n)
  540. size_type log_n;
  541. };
  542. } // end namespace boost
  543. #if defined(BOOST_MSVC)
  544. # pragma warning(pop)
  545. #endif
  546. #endif // BOOST_RELAXED_HEAP_HEADER