code-examples/current/algorithms/heap_binomial.c

256 lines
4.4 KiB
C

#include <limits.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
typedef struct _node {
struct _node* right;
struct _node* child;
long lvl;
long key;
} node;
typedef struct _list {
node *node;
struct _list *next;
} list;
void visit_node_post_order(node *q, void (*callback)(node *)) {
if (!q) {
return;
}
visit_node_post_order(q->child, callback);
visit_node_post_order(q->right, callback);
callback(q);
}
void visit_node_pre_order(node *q, void (*callback)(node *)) {
if (!q) {
return;
}
callback(q);
visit_node_post_order(q->right, callback);
visit_node_post_order(q->child, callback);
}
void node_destroy(list *q) {
void (*func)(node *) = (void (*)(node *))free;
list *t = q;
list *prev = NULL;
while (t) {
visit_node_post_order(t->node, func);
prev = t;
t = t->next;
free(prev);
}
}
node* node_create(long key) {
node *res = (node *)malloc(sizeof(node));
res->lvl = 0;
res->key = key;
res->right = NULL;
res->child = NULL;
return res;
}
node* node_union_same_lvl(node* l, node* r) {
if (!l) {
return r;
}
if (!r) {
return l;
}
if (l->lvl != r->lvl) {
fprintf(stderr, "L lvl %d and R lvl %d do mismatch. Returning NULL\n", l->lvl, r->lvl);
return NULL;
}
node* item;
node* subitem;
if (l->key > r->key) {
// l main
item = l;
subitem = r;
} else {
// r main
item = r;
subitem = l;
}
if (item->child) {
node *t = item->child;
while (t->right != NULL) {
t = t->right;
}
t->right = subitem;
} else {
item->child = subitem;
}
item->lvl++; // we merged two same lvl binomial heap nodes and created on lvl bigger
return item;
}
void dump_node(node *q) {
printf("%d ", q->key);
}
void dump_heap(list *q) {
list *t = q;
while (t) {
printf("Binom lvl %d\n", t->node->lvl);
visit_node_pre_order(t->node, dump_node);
printf("\n");
t = t->next;
}
}
list* node_union(list *q, node *b) {
list *bl,*a = q, *c;
if (!q) {
bl = (list*)malloc(sizeof(list));
bl->node = b;
return bl;
}
int added = 0;
bl = a;
while (a) {
if (a->node->lvl > b->lvl) { // it a always will be bigger or equal
list *el = (list*)malloc(sizeof(list));
el->node = b;
el->next = a;
return el;
} if (a->node->lvl == b->lvl) {
a->node = node_union_same_lvl(a->node, b);
bl = a;
b = a->node;
a = a->next;
added++;
} else {
bl = a;
a = a->next;
}
}
if (!added) {
list *el = (list*)malloc(sizeof(list));
el->node = b;
el->next = NULL;
bl->next = el;
return q;
} else {
return bl;
}
}
list* node_add(list *q, long key) {
node *b = node_create(key);
return node_union(q, b);
}
int node_del(list **q) {
list *t = *q;
long maxValue = INT_MIN;
node *maxTree = NULL;
while (t) {
if (!maxTree || t->node->key > maxValue) {
maxValue = t->node->key;
maxTree = t->node;
}
t = t->next;
}
// we have max element and tree now
t = *q;
if (t->node == maxTree) {
printf("Removed maxtree is the beginning\n");
t = t->next;
} else {
printf("Removed maxtree is not the beginning\n");
while (t && t->next && t->next->node != maxTree) {
t = t->next;
}
t->next = t->next->next;
}
printf("New list pointer 0x%x\n", t);
// we removed the maxTree from the list
node *c = maxTree->child;
node *b;
while (c) {
b = c->right;
c->right = NULL;
t = node_union(t, c);
c = b;
}
*q = t;
return maxValue;
}
const int MAX_BUFFER = 0x1000;
int main() {
char cmd[MAX_BUFFER];
list *q = NULL;
while (1) {
fgets(cmd, MAX_BUFFER, stdin);
printf("Got command %s\n", cmd);
if (strlen(cmd) == 0 || cmd[0] == '\n') {
printf("Finished reading\n");
break;
} else if (memcmp(cmd, "ADD", 3) == 0) {
// handle command add
int x = atoi(cmd + 4);
q = node_add(q, x);
printf("Added number: %d\n", x);
} else if (memcmp(cmd, "DEL", 3) == 0) {
// handle command del
if (!q) {
fprintf(stderr, "There is nothing to delete - heap is empty\n");
continue;
}
long key = node_del(&q);
printf("Removed element: %d\n", key);
} else {
// handle unknown command
fprintf(stderr, "Got unknown command %s\n", cmd);
}
dump_heap(q);
}
if (!q) {
node_destroy(q);
}
return 0;
}