lemon/binom_heap.h
author Peter Kovacs <kpeter@inf.elte.hu>
Thu, 09 Jul 2009 04:07:08 +0200
changeset 750 bb3392fe91f2
parent 748 d1a9224f1e30
child 754 3887d6f994d7
permissions -rw-r--r--
Improve and unify the doc + names in the new heaps (#301)
kpeter@750
     1
/* -*- mode: C++; indent-tabs-mode: nil; -*-
kpeter@748
     2
 *
kpeter@750
     3
 * This file is a part of LEMON, a generic C++ optimization library.
kpeter@748
     4
 *
kpeter@750
     5
 * Copyright (C) 2003-2009
kpeter@748
     6
 * Egervary Jeno Kombinatorikus Optimalizalasi Kutatocsoport
kpeter@748
     7
 * (Egervary Research Group on Combinatorial Optimization, EGRES).
kpeter@748
     8
 *
kpeter@748
     9
 * Permission to use, modify and distribute this software is granted
kpeter@748
    10
 * provided that this copyright notice appears in all copies. For
kpeter@748
    11
 * precise terms see the accompanying LICENSE file.
kpeter@748
    12
 *
kpeter@748
    13
 * This software is provided "AS IS" with no warranty of any kind,
kpeter@748
    14
 * express or implied, and with no claim as to its suitability for any
kpeter@748
    15
 * purpose.
kpeter@748
    16
 *
kpeter@748
    17
 */
kpeter@748
    18
kpeter@748
    19
#ifndef LEMON_BINOM_HEAP_H
kpeter@748
    20
#define LEMON_BINOM_HEAP_H
kpeter@748
    21
kpeter@748
    22
///\file
kpeter@750
    23
///\ingroup heaps
kpeter@748
    24
///\brief Binomial Heap implementation.
kpeter@748
    25
kpeter@748
    26
#include <vector>
kpeter@750
    27
#include <utility>
kpeter@748
    28
#include <functional>
kpeter@748
    29
#include <lemon/math.h>
kpeter@748
    30
#include <lemon/counter.h>
kpeter@748
    31
kpeter@748
    32
namespace lemon {
kpeter@748
    33
kpeter@750
    34
  /// \ingroup heaps
kpeter@748
    35
  ///
kpeter@750
    36
  ///\brief Binomial heap data structure.
kpeter@748
    37
  ///
kpeter@750
    38
  /// This class implements the \e binomial \e heap data structure.
kpeter@750
    39
  /// It fully conforms to the \ref concepts::Heap "heap concept".
kpeter@748
    40
  ///
kpeter@750
    41
  /// The methods \ref increase() and \ref erase() are not efficient
kpeter@750
    42
  /// in a binomial heap. In case of many calls of these operations,
kpeter@750
    43
  /// it is better to use other heap structure, e.g. \ref BinHeap
kpeter@750
    44
  /// "binary heap".
kpeter@748
    45
  ///
kpeter@750
    46
  /// \tparam PR Type of the priorities of the items.
kpeter@750
    47
  /// \tparam IM A read-writable item map with \c int values, used
kpeter@750
    48
  /// internally to handle the cross references.
kpeter@750
    49
  /// \tparam CMP A functor class for comparing the priorities.
kpeter@750
    50
  /// The default is \c std::less<PR>.
kpeter@748
    51
#ifdef DOXYGEN
kpeter@750
    52
  template <typename PR, typename IM, typename CMP>
kpeter@748
    53
#else
kpeter@750
    54
  template <typename PR, typename IM, typename CMP = std::less<PR> >
kpeter@748
    55
#endif
kpeter@748
    56
  class BinomHeap {
kpeter@748
    57
  public:
kpeter@750
    58
    /// Type of the item-int map.
kpeter@750
    59
    typedef IM ItemIntMap;
kpeter@750
    60
    /// Type of the priorities.
kpeter@750
    61
    typedef PR Prio;
kpeter@750
    62
    /// Type of the items stored in the heap.
kpeter@748
    63
    typedef typename ItemIntMap::Key Item;
kpeter@750
    64
    /// Functor type for comparing the priorities.
kpeter@750
    65
    typedef CMP Compare;
kpeter@750
    66
kpeter@750
    67
    /// \brief Type to represent the states of the items.
kpeter@750
    68
    ///
kpeter@750
    69
    /// Each item has a state associated to it. It can be "in heap",
kpeter@750
    70
    /// "pre-heap" or "post-heap". The latter two are indifferent from the
kpeter@750
    71
    /// heap's point of view, but may be useful to the user.
kpeter@750
    72
    ///
kpeter@750
    73
    /// The item-int map must be initialized in such way that it assigns
kpeter@750
    74
    /// \c PRE_HEAP (<tt>-1</tt>) to any element to be put in the heap.
kpeter@750
    75
    enum State {
kpeter@750
    76
      IN_HEAP = 0,    ///< = 0.
kpeter@750
    77
      PRE_HEAP = -1,  ///< = -1.
kpeter@750
    78
      POST_HEAP = -2  ///< = -2.
kpeter@750
    79
    };
kpeter@748
    80
kpeter@748
    81
  private:
kpeter@748
    82
    class store;
kpeter@748
    83
kpeter@750
    84
    std::vector<store> _data;
kpeter@750
    85
    int _min, _head;
kpeter@750
    86
    ItemIntMap &_iim;
kpeter@750
    87
    Compare _comp;
kpeter@750
    88
    int _num_items;
kpeter@748
    89
kpeter@748
    90
  public:
kpeter@750
    91
    /// \brief Constructor.
kpeter@750
    92
    ///
kpeter@750
    93
    /// Constructor.
kpeter@750
    94
    /// \param map A map that assigns \c int values to the items.
kpeter@750
    95
    /// It is used internally to handle the cross references.
kpeter@750
    96
    /// The assigned value must be \c PRE_HEAP (<tt>-1</tt>) for each item.
kpeter@750
    97
    explicit BinomHeap(ItemIntMap &map)
kpeter@750
    98
      : _min(0), _head(-1), _iim(map), _num_items(0) {}
kpeter@748
    99
kpeter@750
   100
    /// \brief Constructor.
kpeter@748
   101
    ///
kpeter@750
   102
    /// Constructor.
kpeter@750
   103
    /// \param map A map that assigns \c int values to the items.
kpeter@750
   104
    /// It is used internally to handle the cross references.
kpeter@750
   105
    /// The assigned value must be \c PRE_HEAP (<tt>-1</tt>) for each item.
kpeter@750
   106
    /// \param comp The function object used for comparing the priorities.
kpeter@750
   107
    BinomHeap(ItemIntMap &map, const Compare &comp)
kpeter@750
   108
      : _min(0), _head(-1), _iim(map), _comp(comp), _num_items(0) {}
kpeter@748
   109
kpeter@748
   110
    /// \brief The number of items stored in the heap.
kpeter@748
   111
    ///
kpeter@750
   112
    /// This function returns the number of items stored in the heap.
kpeter@750
   113
    int size() const { return _num_items; }
kpeter@748
   114
kpeter@750
   115
    /// \brief Check if the heap is empty.
kpeter@748
   116
    ///
kpeter@750
   117
    /// This function returns \c true if the heap is empty.
kpeter@750
   118
    bool empty() const { return _num_items==0; }
kpeter@748
   119
kpeter@750
   120
    /// \brief Make the heap empty.
kpeter@748
   121
    ///
kpeter@750
   122
    /// This functon makes the heap empty.
kpeter@750
   123
    /// It does not change the cross reference map. If you want to reuse
kpeter@750
   124
    /// a heap that is not surely empty, you should first clear it and
kpeter@750
   125
    /// then you should set the cross reference map to \c PRE_HEAP
kpeter@750
   126
    /// for each item.
kpeter@748
   127
    void clear() {
kpeter@750
   128
      _data.clear(); _min=0; _num_items=0; _head=-1;
kpeter@748
   129
    }
kpeter@748
   130
kpeter@750
   131
    /// \brief Set the priority of an item or insert it, if it is
kpeter@750
   132
    /// not stored in the heap.
kpeter@748
   133
    ///
kpeter@750
   134
    /// This method sets the priority of the given item if it is
kpeter@750
   135
    /// already stored in the heap. Otherwise it inserts the given
kpeter@750
   136
    /// item into the heap with the given priority.
kpeter@750
   137
    /// \param item The item.
kpeter@750
   138
    /// \param value The priority.
kpeter@748
   139
    void set (const Item& item, const Prio& value) {
kpeter@750
   140
      int i=_iim[item];
kpeter@750
   141
      if ( i >= 0 && _data[i].in ) {
kpeter@750
   142
        if ( _comp(value, _data[i].prio) ) decrease(item, value);
kpeter@750
   143
        if ( _comp(_data[i].prio, value) ) increase(item, value);
kpeter@748
   144
      } else push(item, value);
kpeter@748
   145
    }
kpeter@748
   146
kpeter@750
   147
    /// \brief Insert an item into the heap with the given priority.
kpeter@748
   148
    ///
kpeter@750
   149
    /// This function inserts the given item into the heap with the
kpeter@750
   150
    /// given priority.
kpeter@750
   151
    /// \param item The item to insert.
kpeter@750
   152
    /// \param value The priority of the item.
kpeter@750
   153
    /// \pre \e item must not be stored in the heap.
kpeter@748
   154
    void push (const Item& item, const Prio& value) {
kpeter@750
   155
      int i=_iim[item];
kpeter@748
   156
      if ( i<0 ) {
kpeter@750
   157
        int s=_data.size();
kpeter@750
   158
        _iim.set( item,s );
kpeter@748
   159
        store st;
kpeter@748
   160
        st.name=item;
kpeter@750
   161
        _data.push_back(st);
kpeter@748
   162
        i=s;
kpeter@748
   163
      }
kpeter@748
   164
      else {
kpeter@750
   165
        _data[i].parent=_data[i].right_neighbor=_data[i].child=-1;
kpeter@750
   166
        _data[i].degree=0;
kpeter@750
   167
        _data[i].in=true;
kpeter@748
   168
      }
kpeter@750
   169
      _data[i].prio=value;
kpeter@748
   170
kpeter@750
   171
      if( 0==_num_items ) { _head=i; _min=i; }
kpeter@748
   172
      else { merge(i); }
kpeter@748
   173
kpeter@750
   174
      _min = findMin();
kpeter@748
   175
kpeter@750
   176
      ++_num_items;
kpeter@748
   177
    }
kpeter@748
   178
kpeter@750
   179
    /// \brief Return the item having minimum priority.
kpeter@748
   180
    ///
kpeter@750
   181
    /// This function returns the item having minimum priority.
kpeter@750
   182
    /// \pre The heap must be non-empty.
kpeter@750
   183
    Item top() const { return _data[_min].name; }
kpeter@748
   184
kpeter@750
   185
    /// \brief The minimum priority.
kpeter@748
   186
    ///
kpeter@750
   187
    /// This function returns the minimum priority.
kpeter@750
   188
    /// \pre The heap must be non-empty.
kpeter@750
   189
    Prio prio() const { return _data[_min].prio; }
kpeter@748
   190
kpeter@750
   191
    /// \brief The priority of the given item.
kpeter@748
   192
    ///
kpeter@750
   193
    /// This function returns the priority of the given item.
kpeter@750
   194
    /// \param item The item.
kpeter@750
   195
    /// \pre \e item must be in the heap.
kpeter@748
   196
    const Prio& operator[](const Item& item) const {
kpeter@750
   197
      return _data[_iim[item]].prio;
kpeter@748
   198
    }
kpeter@748
   199
kpeter@750
   200
    /// \brief Remove the item having minimum priority.
kpeter@748
   201
    ///
kpeter@750
   202
    /// This function removes the item having minimum priority.
kpeter@748
   203
    /// \pre The heap must be non-empty.
kpeter@748
   204
    void pop() {
kpeter@750
   205
      _data[_min].in=false;
kpeter@748
   206
kpeter@748
   207
      int head_child=-1;
kpeter@750
   208
      if ( _data[_min].child!=-1 ) {
kpeter@750
   209
        int child=_data[_min].child;
kpeter@748
   210
        int neighb;
kpeter@748
   211
        int prev=-1;
kpeter@748
   212
        while( child!=-1 ) {
kpeter@750
   213
          neighb=_data[child].right_neighbor;
kpeter@750
   214
          _data[child].parent=-1;
kpeter@750
   215
          _data[child].right_neighbor=prev;
kpeter@748
   216
          head_child=child;
kpeter@748
   217
          prev=child;
kpeter@748
   218
          child=neighb;
kpeter@748
   219
        }
kpeter@748
   220
      }
kpeter@748
   221
kpeter@748
   222
      // The first case is that there are only one root.
kpeter@750
   223
      if ( -1==_data[_head].right_neighbor ) {
kpeter@750
   224
        _head=head_child;
kpeter@748
   225
      }
kpeter@748
   226
      // The case where there are more roots.
kpeter@748
   227
      else {
kpeter@750
   228
        if( _head!=_min )  { unlace(_min); }
kpeter@750
   229
        else { _head=_data[_head].right_neighbor; }
kpeter@748
   230
kpeter@748
   231
        merge(head_child);
kpeter@748
   232
      }
kpeter@750
   233
      _min=findMin();
kpeter@750
   234
      --_num_items;
kpeter@748
   235
    }
kpeter@748
   236
kpeter@750
   237
    /// \brief Remove the given item from the heap.
kpeter@748
   238
    ///
kpeter@750
   239
    /// This function removes the given item from the heap if it is
kpeter@750
   240
    /// already stored.
kpeter@750
   241
    /// \param item The item to delete.
kpeter@750
   242
    /// \pre \e item must be in the heap.
kpeter@748
   243
    void erase (const Item& item) {
kpeter@750
   244
      int i=_iim[item];
kpeter@750
   245
      if ( i >= 0 && _data[i].in ) {
kpeter@750
   246
        decrease( item, _data[_min].prio-1 );
kpeter@748
   247
        pop();
kpeter@748
   248
      }
kpeter@748
   249
    }
kpeter@748
   250
kpeter@750
   251
    /// \brief Decrease the priority of an item to the given value.
kpeter@748
   252
    ///
kpeter@750
   253
    /// This function decreases the priority of an item to the given value.
kpeter@750
   254
    /// \param item The item.
kpeter@750
   255
    /// \param value The priority.
kpeter@750
   256
    /// \pre \e item must be stored in the heap with priority at least \e value.
kpeter@748
   257
    void decrease (Item item, const Prio& value) {
kpeter@750
   258
      int i=_iim[item];
kpeter@748
   259
kpeter@750
   260
      if( _comp( value,_data[i].prio ) ) {
kpeter@750
   261
        _data[i].prio=value;
kpeter@748
   262
kpeter@750
   263
        int p_loc=_data[i].parent, loc=i;
kpeter@748
   264
        int parent, child, neighb;
kpeter@748
   265
kpeter@750
   266
        while( -1!=p_loc && _comp(_data[loc].prio,_data[p_loc].prio) ) {
kpeter@748
   267
kpeter@748
   268
          // parent set for other loc_child
kpeter@750
   269
          child=_data[loc].child;
kpeter@748
   270
          while( -1!=child ) {
kpeter@750
   271
            _data[child].parent=p_loc;
kpeter@750
   272
            child=_data[child].right_neighbor;
kpeter@748
   273
          }
kpeter@748
   274
kpeter@748
   275
          // parent set for other p_loc_child
kpeter@750
   276
          child=_data[p_loc].child;
kpeter@748
   277
          while( -1!=child ) {
kpeter@750
   278
            _data[child].parent=loc;
kpeter@750
   279
            child=_data[child].right_neighbor;
kpeter@748
   280
          }
kpeter@748
   281
kpeter@750
   282
          child=_data[p_loc].child;
kpeter@750
   283
          _data[p_loc].child=_data[loc].child;
kpeter@748
   284
          if( child==loc )
kpeter@748
   285
            child=p_loc;
kpeter@750
   286
          _data[loc].child=child;
kpeter@748
   287
kpeter@748
   288
          // left_neighb set for p_loc
kpeter@750
   289
          if( _data[loc].child!=p_loc ) {
kpeter@750
   290
            while( _data[child].right_neighbor!=loc )
kpeter@750
   291
              child=_data[child].right_neighbor;
kpeter@750
   292
            _data[child].right_neighbor=p_loc;
kpeter@748
   293
          }
kpeter@748
   294
kpeter@748
   295
          // left_neighb set for loc
kpeter@750
   296
          parent=_data[p_loc].parent;
kpeter@750
   297
          if( -1!=parent ) child=_data[parent].child;
kpeter@750
   298
          else child=_head;
kpeter@748
   299
kpeter@748
   300
          if( child!=p_loc ) {
kpeter@750
   301
            while( _data[child].right_neighbor!=p_loc )
kpeter@750
   302
              child=_data[child].right_neighbor;
kpeter@750
   303
            _data[child].right_neighbor=loc;
kpeter@748
   304
          }
kpeter@748
   305
kpeter@750
   306
          neighb=_data[p_loc].right_neighbor;
kpeter@750
   307
          _data[p_loc].right_neighbor=_data[loc].right_neighbor;
kpeter@750
   308
          _data[loc].right_neighbor=neighb;
kpeter@748
   309
kpeter@750
   310
          _data[p_loc].parent=loc;
kpeter@750
   311
          _data[loc].parent=parent;
kpeter@748
   312
kpeter@750
   313
          if( -1!=parent && _data[parent].child==p_loc )
kpeter@750
   314
            _data[parent].child=loc;
kpeter@748
   315
kpeter@748
   316
          /*if new parent will be the first root*/
kpeter@750
   317
          if( _head==p_loc )
kpeter@750
   318
            _head=loc;
kpeter@748
   319
kpeter@750
   320
          p_loc=_data[loc].parent;
kpeter@748
   321
        }
kpeter@748
   322
      }
kpeter@750
   323
      if( _comp(value,_data[_min].prio) ) {
kpeter@750
   324
        _min=i;
kpeter@748
   325
      }
kpeter@748
   326
    }
kpeter@748
   327
kpeter@750
   328
    /// \brief Increase the priority of an item to the given value.
kpeter@748
   329
    ///
kpeter@750
   330
    /// This function increases the priority of an item to the given value.
kpeter@750
   331
    /// \param item The item.
kpeter@750
   332
    /// \param value The priority.
kpeter@750
   333
    /// \pre \e item must be stored in the heap with priority at most \e value.
kpeter@748
   334
    void increase (Item item, const Prio& value) {
kpeter@748
   335
      erase(item);
kpeter@748
   336
      push(item, value);
kpeter@748
   337
    }
kpeter@748
   338
kpeter@750
   339
    /// \brief Return the state of an item.
kpeter@748
   340
    ///
kpeter@750
   341
    /// This method returns \c PRE_HEAP if the given item has never
kpeter@750
   342
    /// been in the heap, \c IN_HEAP if it is in the heap at the moment,
kpeter@750
   343
    /// and \c POST_HEAP otherwise.
kpeter@750
   344
    /// In the latter case it is possible that the item will get back
kpeter@750
   345
    /// to the heap again.
kpeter@750
   346
    /// \param item The item.
kpeter@748
   347
    State state(const Item &item) const {
kpeter@750
   348
      int i=_iim[item];
kpeter@748
   349
      if( i>=0 ) {
kpeter@750
   350
        if ( _data[i].in ) i=0;
kpeter@748
   351
        else i=-2;
kpeter@748
   352
      }
kpeter@748
   353
      return State(i);
kpeter@748
   354
    }
kpeter@748
   355
kpeter@750
   356
    /// \brief Set the state of an item in the heap.
kpeter@748
   357
    ///
kpeter@750
   358
    /// This function sets the state of the given item in the heap.
kpeter@750
   359
    /// It can be used to manually clear the heap when it is important
kpeter@750
   360
    /// to achive better time complexity.
kpeter@748
   361
    /// \param i The item.
kpeter@748
   362
    /// \param st The state. It should not be \c IN_HEAP.
kpeter@748
   363
    void state(const Item& i, State st) {
kpeter@748
   364
      switch (st) {
kpeter@748
   365
      case POST_HEAP:
kpeter@748
   366
      case PRE_HEAP:
kpeter@748
   367
        if (state(i) == IN_HEAP) {
kpeter@748
   368
          erase(i);
kpeter@748
   369
        }
kpeter@750
   370
        _iim[i] = st;
kpeter@748
   371
        break;
kpeter@748
   372
      case IN_HEAP:
kpeter@748
   373
        break;
kpeter@748
   374
      }
kpeter@748
   375
    }
kpeter@748
   376
kpeter@748
   377
  private:
kpeter@750
   378
    int findMin() {
kpeter@748
   379
      int min_loc=-1, min_val;
kpeter@750
   380
      int x=_head;
kpeter@748
   381
      if( x!=-1 ) {
kpeter@750
   382
        min_val=_data[x].prio;
kpeter@748
   383
        min_loc=x;
kpeter@750
   384
        x=_data[x].right_neighbor;
kpeter@748
   385
kpeter@748
   386
        while( x!=-1 ) {
kpeter@750
   387
          if( _comp( _data[x].prio,min_val ) ) {
kpeter@750
   388
            min_val=_data[x].prio;
kpeter@748
   389
            min_loc=x;
kpeter@748
   390
          }
kpeter@750
   391
          x=_data[x].right_neighbor;
kpeter@748
   392
        }
kpeter@748
   393
      }
kpeter@748
   394
      return min_loc;
kpeter@748
   395
    }
kpeter@748
   396
kpeter@748
   397
    void merge(int a) {
kpeter@748
   398
      interleave(a);
kpeter@748
   399
kpeter@750
   400
      int x=_head;
kpeter@748
   401
      if( -1!=x ) {
kpeter@750
   402
        int x_prev=-1, x_next=_data[x].right_neighbor;
kpeter@748
   403
        while( -1!=x_next ) {
kpeter@750
   404
          if( _data[x].degree!=_data[x_next].degree || ( -1!=_data[x_next].right_neighbor && _data[_data[x_next].right_neighbor].degree==_data[x].degree ) ) {
kpeter@748
   405
            x_prev=x;
kpeter@748
   406
            x=x_next;
kpeter@748
   407
          }
kpeter@748
   408
          else {
kpeter@750
   409
            if( _comp(_data[x].prio,_data[x_next].prio) ) {
kpeter@750
   410
              _data[x].right_neighbor=_data[x_next].right_neighbor;
kpeter@748
   411
              fuse(x_next,x);
kpeter@748
   412
            }
kpeter@748
   413
            else {
kpeter@750
   414
              if( -1==x_prev ) { _head=x_next; }
kpeter@748
   415
              else {
kpeter@750
   416
                _data[x_prev].right_neighbor=x_next;
kpeter@748
   417
              }
kpeter@748
   418
              fuse(x,x_next);
kpeter@748
   419
              x=x_next;
kpeter@748
   420
            }
kpeter@748
   421
          }
kpeter@750
   422
          x_next=_data[x].right_neighbor;
kpeter@748
   423
        }
kpeter@748
   424
      }
kpeter@748
   425
    }
kpeter@748
   426
kpeter@748
   427
    void interleave(int a) {
kpeter@748
   428
      int other=-1, head_other=-1;
kpeter@748
   429
kpeter@750
   430
      while( -1!=a || -1!=_head ) {
kpeter@748
   431
        if( -1==a ) {
kpeter@748
   432
          if( -1==head_other ) {
kpeter@750
   433
            head_other=_head;
kpeter@748
   434
          }
kpeter@748
   435
          else {
kpeter@750
   436
            _data[other].right_neighbor=_head;
kpeter@748
   437
          }
kpeter@750
   438
          _head=-1;
kpeter@748
   439
        }
kpeter@750
   440
        else if( -1==_head ) {
kpeter@748
   441
          if( -1==head_other ) {
kpeter@748
   442
            head_other=a;
kpeter@748
   443
          }
kpeter@748
   444
          else {
kpeter@750
   445
            _data[other].right_neighbor=a;
kpeter@748
   446
          }
kpeter@748
   447
          a=-1;
kpeter@748
   448
        }
kpeter@748
   449
        else {
kpeter@750
   450
          if( _data[a].degree<_data[_head].degree ) {
kpeter@748
   451
            if( -1==head_other ) {
kpeter@748
   452
              head_other=a;
kpeter@748
   453
            }
kpeter@748
   454
            else {
kpeter@750
   455
              _data[other].right_neighbor=a;
kpeter@748
   456
            }
kpeter@748
   457
            other=a;
kpeter@750
   458
            a=_data[a].right_neighbor;
kpeter@748
   459
          }
kpeter@748
   460
          else {
kpeter@748
   461
            if( -1==head_other ) {
kpeter@750
   462
              head_other=_head;
kpeter@748
   463
            }
kpeter@748
   464
            else {
kpeter@750
   465
              _data[other].right_neighbor=_head;
kpeter@748
   466
            }
kpeter@750
   467
            other=_head;
kpeter@750
   468
            _head=_data[_head].right_neighbor;
kpeter@748
   469
          }
kpeter@748
   470
        }
kpeter@748
   471
      }
kpeter@750
   472
      _head=head_other;
kpeter@748
   473
    }
kpeter@748
   474
kpeter@748
   475
    // Lacing a under b
kpeter@748
   476
    void fuse(int a, int b) {
kpeter@750
   477
      _data[a].parent=b;
kpeter@750
   478
      _data[a].right_neighbor=_data[b].child;
kpeter@750
   479
      _data[b].child=a;
kpeter@748
   480
kpeter@750
   481
      ++_data[b].degree;
kpeter@748
   482
    }
kpeter@748
   483
kpeter@748
   484
    // It is invoked only if a has siblings.
kpeter@748
   485
    void unlace(int a) {
kpeter@750
   486
      int neighb=_data[a].right_neighbor;
kpeter@750
   487
      int other=_head;
kpeter@748
   488
kpeter@750
   489
      while( _data[other].right_neighbor!=a )
kpeter@750
   490
        other=_data[other].right_neighbor;
kpeter@750
   491
      _data[other].right_neighbor=neighb;
kpeter@748
   492
    }
kpeter@748
   493
kpeter@748
   494
  private:
kpeter@748
   495
kpeter@748
   496
    class store {
kpeter@748
   497
      friend class BinomHeap;
kpeter@748
   498
kpeter@748
   499
      Item name;
kpeter@748
   500
      int parent;
kpeter@748
   501
      int right_neighbor;
kpeter@748
   502
      int child;
kpeter@748
   503
      int degree;
kpeter@748
   504
      bool in;
kpeter@748
   505
      Prio prio;
kpeter@748
   506
kpeter@748
   507
      store() : parent(-1), right_neighbor(-1), child(-1), degree(0), in(true) {}
kpeter@748
   508
    };
kpeter@748
   509
  };
kpeter@748
   510
kpeter@748
   511
} //namespace lemon
kpeter@748
   512
kpeter@748
   513
#endif //LEMON_BINOM_HEAP_H
kpeter@748
   514