Blame src/xheap.h

Packit 8f70b4
/*
Packit 8f70b4
 * lftp - file transfer program
Packit 8f70b4
 *
Packit 8f70b4
 * Copyright (c) 1996-2013 by Alexander V. Lukyanov (lav@yars.free.net)
Packit 8f70b4
 *
Packit 8f70b4
 * This program is free software; you can redistribute it and/or modify
Packit 8f70b4
 * it under the terms of the GNU General Public License as published by
Packit 8f70b4
 * the Free Software Foundation; either version 3 of the License, or
Packit 8f70b4
 * (at your option) any later version.
Packit 8f70b4
 *
Packit 8f70b4
 * This program is distributed in the hope that it will be useful,
Packit 8f70b4
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 8f70b4
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 8f70b4
 * GNU General Public License for more details.
Packit 8f70b4
 *
Packit 8f70b4
 * You should have received a copy of the GNU General Public License
Packit 8f70b4
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
Packit 8f70b4
 */
Packit 8f70b4
Packit 8f70b4
#ifndef XHEAP_H
Packit 8f70b4
#define XHEAP_H 1
Packit 8f70b4
Packit 8f70b4
// min-heap implementation with random remove
Packit 8f70b4
Packit 8f70b4
#include <assert.h>
Packit 8f70b4
#include "xarray.h"
Packit 8f70b4
Packit 8f70b4
#define costly_assert(x) /*assert(x)*/
Packit 8f70b4
Packit 8f70b4
template<class T>
Packit 8f70b4
class xheap
Packit 8f70b4
{
Packit 8f70b4
public:
Packit 8f70b4
   class node
Packit 8f70b4
   {
Packit 8f70b4
      T *obj;
Packit 8f70b4
      int heap_index;
Packit 8f70b4
      friend class xheap<T>;
Packit 8f70b4
   public:
Packit 8f70b4
      node(T *t) : obj(t), heap_index(0) {}
Packit 8f70b4
   };
Packit 8f70b4
Packit 8f70b4
private:
Packit 8f70b4
   xarray<node*> heap;
Packit 8f70b4
Packit 8f70b4
   int count() const { return heap.count(); }
Packit 8f70b4
   node*& ptr(int i) { return heap[i-1]; }
Packit 8f70b4
   T& elem(int i) { return *(ptr(i)->obj); }
Packit 8f70b4
   void swap(int a,int b) {
Packit 8f70b4
      ptr(a)=replace_value(ptr(b),ptr(a));
Packit 8f70b4
      ptr(a)->heap_index=a;
Packit 8f70b4
      ptr(b)->heap_index=b;
Packit 8f70b4
   }
Packit 8f70b4
   void siftup(int i) {
Packit 8f70b4
      while(i>1 && elem(i)
Packit 8f70b4
	 swap(i,i/2);
Packit 8f70b4
	 i/=2;
Packit 8f70b4
      }
Packit 8f70b4
   }
Packit 8f70b4
   void siftdown(int i) {
Packit 8f70b4
      while(i<=count()/2) {
Packit 8f70b4
	 int i2=i*2;
Packit 8f70b4
	 // choose smallest of heirs
Packit 8f70b4
	 if(i2
Packit 8f70b4
	    i2++;
Packit 8f70b4
	 if(elem(i)
Packit 8f70b4
	    break;
Packit 8f70b4
	 swap(i,i2);
Packit 8f70b4
	 i=i2;
Packit 8f70b4
      }
Packit 8f70b4
   }
Packit 8f70b4
   bool is_heap(int a,int b) {
Packit 8f70b4
      while(a*2<=b) {
Packit 8f70b4
	 if(elem(a*2)
Packit 8f70b4
	    return false;
Packit 8f70b4
	 a++;
Packit 8f70b4
      }
Packit 8f70b4
      return true;
Packit 8f70b4
   }
Packit 8f70b4
   void chop() {
Packit 8f70b4
      ptr(count())->heap_index=0;
Packit 8f70b4
      heap.chop();
Packit 8f70b4
   }
Packit 8f70b4
   void fix(int i) {
Packit 8f70b4
      siftdown(i);
Packit 8f70b4
      siftup(i);
Packit 8f70b4
   }
Packit 8f70b4
   void remove(int i) {
Packit 8f70b4
      if(i==count()) {
Packit 8f70b4
	 chop();
Packit 8f70b4
	 return;
Packit 8f70b4
      }
Packit 8f70b4
      assert(i>0 && i
Packit 8f70b4
      swap(i,count());
Packit 8f70b4
      chop();
Packit 8f70b4
      fix(i);
Packit 8f70b4
      costly_assert(is_heap(1,count()));
Packit 8f70b4
   }
Packit 8f70b4
public:
Packit 8f70b4
   void add(node& n) {
Packit 8f70b4
      if(n.heap_index) {
Packit 8f70b4
	 int i=n.heap_index;
Packit 8f70b4
	 assert(i>0 && i<=count());
Packit 8f70b4
	 assert(ptr(i)==&n);
Packit 8f70b4
	 return;
Packit 8f70b4
      }
Packit 8f70b4
      heap.append(&n);
Packit 8f70b4
      siftup(n.heap_index=count());
Packit 8f70b4
      costly_assert(is_heap(1,count()));
Packit 8f70b4
   }
Packit 8f70b4
   void fix(node& n) {
Packit 8f70b4
      fix(n.heap_index);
Packit 8f70b4
   }
Packit 8f70b4
   T *get_min() {
Packit 8f70b4
      return count()>0?ptr(1)->obj:0;
Packit 8f70b4
   }
Packit 8f70b4
   T *pop_min() {
Packit 8f70b4
      T *m=get_min();
Packit 8f70b4
      if(!m)
Packit 8f70b4
	 return 0;
Packit 8f70b4
      remove(1);
Packit 8f70b4
      return m;
Packit 8f70b4
   }
Packit 8f70b4
   void remove(node& x) {
Packit 8f70b4
      if(!x.heap_index)
Packit 8f70b4
	 return;
Packit 8f70b4
      assert(ptr(x.heap_index)==&x);
Packit 8f70b4
      remove(x.heap_index);
Packit 8f70b4
      assert(!x.heap_index);
Packit 8f70b4
   }
Packit 8f70b4
};
Packit 8f70b4
Packit 8f70b4
#endif//XHEAP_H