|
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
|