diff --git a/current/algorithms/heap_binomial.c b/current/algorithms/heap_binomial.c new file mode 100644 index 0000000..c6f8215 --- /dev/null +++ b/current/algorithms/heap_binomial.c @@ -0,0 +1,191 @@ +#include +#include +#include + +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; + while (t) { + visit_node_post_order(t->node, func); + t = t->next; + } +} + +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_add(list *q, long key) { + list *c = NULL; + list *a = q; + node *b = node_create(key); + list *bl; + + if (!q) { + a = (list*)malloc(sizeof(list)); + a->node = b; + return 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; + } else { + c = a->next; + a->node = node_union_same_lvl(a->node, b); + bl = a; + b = a->node; + a = c; + } + } + + return bl; + +} + +int node_del(node *q) { + // TODO + return -1; +} + + +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; + } + // TODO: 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; +} diff --git a/current/algorithms/heap_binomial.c.old b/current/algorithms/heap_binomial.c.old new file mode 100644 index 0000000..a334858 --- /dev/null +++ b/current/algorithms/heap_binomial.c.old @@ -0,0 +1,157 @@ +#include +#include +#include + +typedef struct _node { + struct _node* left; + struct _node* right; + struct _node* child; + long lvl; + long key; +} node; + +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 node_destroy(node *q) { + void (*func)(node *) = (void (*)(node *))free; + visit_node_post_order(q, func); +} + +node* node_create(long key) { + node *res = (node *)malloc(sizeof(node)); + res->lvl = 0; + res->key = key; + res->left = res; + res->right = res; + 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) { + item->child->left->right = subitem; // we have two directional list (left,right links) where left of the beginning is the + // end and right of the end is the beginning + } else { + item->child = subitem; + } + item->lvl++; // we merged two same lvl binomial heap nodes and created on lvl bigger + + return item; +} + +node* node_add(node *q, long key) { + node *c = NULL; + node *a = q; + node *b = node_create(key); + + if (q->lvl < 0) { + return b; + } + + while (a) { + if (a->lvl > b->lvl) { // it a always will be bigger or equal + b->right = a; + a->left->right = b; + b->left = a->left; // a's ending + a->left = b; + return b; + } else { + b = node_union_same_lvl(a, b); + c = a->right; + a->right = NULL; + + if (c == q) { + break; + } + + a = c; + } + } + + return b; + +} + +int node_del(node *q) { + // TODO + return -1; +} + + +const int MAX_BUFFER = 0x1000; + +int main() { + char cmd[MAX_BUFFER]; + node *q = (node *)malloc(sizeof(node)); + q->lvl = -1; + q->key = 0xdeadbeaf; // it doesn't mother honestly + 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); + } + + } + + if (!q) { + node_destroy(q); + } + + return 0; +}