home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
Frozen Fish 1: Amiga
/
FrozenFish-Apr94.iso
/
bbs
/
alib
/
d1xx
/
d103
/
avltrees.lha
/
AvlTrees
/
tree.c
< prev
next >
Wrap
C/C++ Source or Header
|
1987-10-22
|
10KB
|
512 lines
/* as_tree - tree library for as
* vix 14dec85 [written]
* vix 02feb86 [added tree balancing from wirth "a+ds=p" p. 220-221]
* vix 06feb86 [added tree_mung()]
* vix 20jun86 [added tree_delete per wirth a+ds (mod2 v.) p. 224]
* vix 23jun86 [added delete uar to add for replaced nodes]
*/
/* This program text was created by Paul Vixie using examples from the book:
* "Algorithms & Data Structures," Niklaus Wirth, Prentice-Hall, 1986, ISBN
* 0-13-022005-1. This code and associated documentation is hereby placed
* in the public domain.
*/
/*#define DEBUG "tree"*/
#include <stdio.h>
#include "vixie.h"
#include "tree.h"
#ifdef DEBUG
#define MSG(msg) printf("DEBUG: '%s'\n", msg);
#else
#define MSG(msg)
#endif
void tree_init(ppr_tree)
tree **ppr_tree;
{
ENTER("tree_init")
*ppr_tree = NULL;
EXITV
}
char *tree_srch(ppr_tree, pfi_compare, pc_user)
tree **ppr_tree;
int (*pfi_compare)();
char *pc_user;
{
register int i_comp;
register tree *pr_new;
ENTER("tree_srch")
if (*ppr_tree)
{
i_comp = (*pfi_compare)(pc_user, (**ppr_tree).tree_p);
if (i_comp > 0)
EXIT(tree_srch(
&(**ppr_tree).tree_r,
pfi_compare,
pc_user
))
if (i_comp < 0)
EXIT(tree_srch(
&(**ppr_tree).tree_l,
pfi_compare,
pc_user
))
/* not higher, not lower... this must be the one.
*/
EXIT((**ppr_tree).tree_p)
}
/* grounded. NOT found.
*/
EXIT(NULL)
}
void tree_add(ppr_tree, pfi_compare, pc_user, pfi_delete)
tree **ppr_tree;
int (*pfi_compare)();
char *pc_user;
int (*pfi_delete)();
{
void sprout();
int i_balance = FALSE;
ENTER("tree_add")
sprout(ppr_tree, pc_user, &i_balance, pfi_compare, pfi_delete);
EXITV
}
static void sprout(ppr, pc_data, pi_balance, pfi_compare, pfi_delete)
tree **ppr;
char *pc_data;
int *pi_balance;
int (*pfi_compare)();
int (*pfi_delete)();
{
tree *p1, *p2;
int cmp;
ENTER("sprout")
/* are we grounded? if so, add the node "here" and set the rebalance
* flag, then exit.
*/
if (!*ppr) {
MSG("grounded. adding new node, setting h=true")
*ppr = (tree *) malloc(sizeof(tree));
(*ppr)->tree_l = NULL;
(*ppr)->tree_r = NULL;
(*ppr)->tree_b = 0;
(*ppr)->tree_p = pc_data;
*pi_balance = TRUE;
EXITV
}
/* compare the data using routine passed by caller.
*/
cmp = (*pfi_compare)(pc_data, (*ppr)->tree_p);
/* if LESS, prepare to move to the left.
*/
if (cmp < 0) {
MSG("LESS. sprouting left.")
sprout(&(*ppr)->tree_l, pc_data, pi_balance,
pfi_compare, pfi_delete);
if (*pi_balance) { /* left branch has grown longer */
MSG("LESS: left branch has grown")
switch ((*ppr)->tree_b)
{
case 1: /* right branch WAS longer; balance is ok now */
MSG("LESS: case 1.. balnce restored implicitly")
(*ppr)->tree_b = 0;
*pi_balance = FALSE;
break;
case 0: /* balance WAS okay; now left branch longer */
MSG("LESS: case 0.. balnce bad but still ok")
(*ppr)->tree_b = -1;
break;
case -1:
/* left branch was already too long. rebalnce */
MSG("LESS: case -1: rebalancing")
p1 = (*ppr)->tree_l;
if (p1->tree_b == -1) { /* LL */
MSG("LESS: single LL")
(*ppr)->tree_l = p1->tree_r;
p1->tree_r = *ppr;
(*ppr)->tree_b = 0;
*ppr = p1;
}
else { /* double LR */
MSG("LESS: double LR")
p2 = p1->tree_r;
p1->tree_r = p2->tree_l;
p2->tree_l = p1;
(*ppr)->tree_l = p2->tree_r;
p2->tree_r = *ppr;
if (p2->tree_b == -1)
(*ppr)->tree_b = 1;
else
(*ppr)->tree_b = 0;
if (p2->tree_b == 1)
p1->tree_b = -1;
else
p1->tree_b = 0;
*ppr = p2;
} /*else*/
(*ppr)->tree_b = 0;
*pi_balance = FALSE;
} /*switch*/
} /*if*/
EXITV
} /*if*/
/* if MORE, prepare to move to the right.
*/
if (cmp > 0) {
MSG("MORE: sprouting to the right")
sprout(&(*ppr)->tree_r, pc_data, pi_balance,
pfi_compare, pfi_delete);
if (*pi_balance) { /* right branch has grown longer */
MSG("MORE: right branch has grown")
switch ((*ppr)->tree_b)
{
case -1:MSG("MORE: balance was off, fixed implicitly")
(*ppr)->tree_b = 0;
*pi_balance = FALSE;
break;
case 0: MSG("MORE: balance was okay, now off but ok")
(*ppr)->tree_b = 1;
break;
case 1: MSG("MORE: balance was off, need to rebalance")
p1 = (*ppr)->tree_r;
if (p1->tree_b == 1) { /* RR */
MSG("MORE: single RR")
(*ppr)->tree_r = p1->tree_l;
p1->tree_l = *ppr;
(*ppr)->tree_b = 0;
*ppr = p1;
}
else { /* double RL */
MSG("MORE: double RL")
p2 = p1->tree_l;
p1->tree_l = p2->tree_r;
p2->tree_r = p1;
(*ppr)->tree_r = p2->tree_l;
p2->tree_l = *ppr;
if (p2->tree_b == 1)
(*ppr)->tree_b = -1;
else
(*ppr)->tree_b = 0;
if (p2->tree_b == -1)
p1->tree_b = 1;
else
p1->tree_b = 0;
*ppr = p2;
} /*else*/
(*ppr)->tree_b = 0;
*pi_balance = FALSE;
} /*switch*/
} /*if*/
EXITV
} /*if*/
/* not less, not more: this is the same key! replace...
*/
MSG("I found it! Replacing data value")
*pi_balance = FALSE;
if (pfi_delete)
(*pfi_delete)((*ppr)->tree_p);
(*ppr)->tree_p = pc_data;
EXITV
}
int tree_delete(ppr_p, pfi_compare, pc_user, pfi_uar)
tree **ppr_p;
int (*pfi_compare)();
char *pc_user;
int (*pfi_uar)();
{
int i_balance = FALSE, i_uar_called = FALSE;
ENTER("tree_delete");
EXIT(delete(ppr_p, pfi_compare, pc_user, pfi_uar,
&i_balance, &i_uar_called))
}
static int delete(ppr_p, pfi_compare, pc_user, pfi_uar,
pi_balance, pi_uar_called)
tree **ppr_p;
int (*pfi_compare)();
char *pc_user;
int (*pfi_uar)();
int *pi_balance;
int *pi_uar_called;
{
void del(), balanceL(), balanceR();
tree *pr_q;
int i_comp, i_ret;
ENTER("delete")
if (*ppr_p == NULL) {
MSG("key not in tree")
EXIT(FALSE)
}
i_comp = (*pfi_compare)((*ppr_p)->tree_p, pc_user);
if (i_comp > 0) {
MSG("too high - scan left")
i_ret = delete(&(*ppr_p)->tree_l, pfi_compare, pc_user, pfi_uar,
pi_balance, pi_uar_called);
if (*pi_balance)
balanceL(ppr_p, pi_balance);
}
else if (i_comp < 0) {
MSG("too low - scan right")
i_ret = delete(&(*ppr_p)->tree_r, pfi_compare, pc_user, pfi_uar,
pi_balance, pi_uar_called);
if (*pi_balance)
balanceR(ppr_p, pi_balance);
}
else {
MSG("equal")
pr_q = *ppr_p;
if (pr_q->tree_r == NULL) {
MSG("right subtree null")
*ppr_p = pr_q->tree_l;
*pi_balance = TRUE;
}
else if (pr_q->tree_l == NULL) {
MSG("right subtree non-null, left subtree null")
*ppr_p = pr_q->tree_r;
*pi_balance = TRUE;
}
else {
MSG("neither subtree null")
del(&pr_q->tree_l, pi_balance, &pr_q, pfi_uar,
pi_uar_called);
if (*pi_balance)
balanceL(ppr_p, pi_balance);
}
free(pr_q);
if (!*pi_uar_called && pfi_uar)
(*pfi_uar)(pr_q->tree_p);
i_ret = TRUE;
}
EXIT(i_ret)
}
static void del(ppr_r, pi_balance, ppr_q, pfi_uar, pi_uar_called)
tree **ppr_r;
int *pi_balance;
tree **ppr_q;
int (*pfi_uar)();
int *pi_uar_called;
{
void balanceR();
ENTER("del")
if ((*ppr_r)->tree_r != NULL) {
del(&(*ppr_r)->tree_r, pi_balance, ppr_q, pfi_uar,
pi_uar_called);
if (*pi_balance)
balanceR(ppr_r, pi_balance);
} else {
if (pfi_uar)
(*pfi_uar)((*ppr_q)->tree_p);
*pi_uar_called = TRUE;
(*ppr_q)->tree_p = (*ppr_r)->tree_p;
*ppr_q = *ppr_r;
*ppr_r = (*ppr_r)->tree_l;
*pi_balance = TRUE;
}
EXITV
}
static void balanceL(ppr_p, pi_balance)
tree **ppr_p;
int *pi_balance;
{
tree *p1, *p2;
int b1, b2;
ENTER("balanceL")
MSG("left branch has shrunk")
switch ((*ppr_p)->tree_b)
{
case -1: MSG("was imbalanced, fixed implicitly")
(*ppr_p)->tree_b = 0;
break;
case 0: MSG("was okay, is now one off")
(*ppr_p)->tree_b = 1;
*pi_balance = FALSE;
break;
case 1: MSG("was already off, this is too much")
p1 = (*ppr_p)->tree_r;
b1 = p1->tree_b;
if (b1 >= 0) {
MSG("single RR")
(*ppr_p)->tree_r = p1->tree_l;
p1->tree_l = *ppr_p;
if (b1 == 0) {
MSG("b1 == 0")
(*ppr_p)->tree_b = 1;
p1->tree_b = -1;
*pi_balance = FALSE;
} else {
MSG("b1 != 0")
(*ppr_p)->tree_b = 0;
p1->tree_b = 0;
}
*ppr_p = p1;
} else {
MSG("double RL")
p2 = p1->tree_l;
b2 = p2->tree_b;
p1->tree_l = p2->tree_r;
p2->tree_r = p1;
(*ppr_p)->tree_r = p2->tree_l;
p2->tree_l = *ppr_p;
if (b2 == 1)
(*ppr_p)->tree_b = -1;
else
(*ppr_p)->tree_b = 0;
if (b2 == -1)
p1->tree_b = 1;
else
p1->tree_b = 0;
*ppr_p = p2;
p2->tree_b = 0;
}
}
EXITV
}
static void balanceR(ppr_p, pi_balance)
tree **ppr_p;
int *pi_balance;
{
tree *p1, *p2;
int b1, b2;
ENTER("balanceR")
MSG("right branch has shrunk")
switch ((*ppr_p)->tree_b)
{
case 1: MSG("was imbalanced, fixed implicitly")
(*ppr_p)->tree_b = 0;
break;
case 0: MSG("was okay, is now one off")
(*ppr_p)->tree_b = -1;
*pi_balance = FALSE;
break;
case -1: MSG("was already off, this is too much")
p1 = (*ppr_p)->tree_l;
b1 = p1->tree_b;
if (b1 <= 0) {
MSG("single LL")
(*ppr_p)->tree_l = p1->tree_r;
p1->tree_r = *ppr_p;
if (b1 == 0) {
MSG("b1 == 0")
(*ppr_p)->tree_b = -1;
p1->tree_b = 1;
*pi_balance = FALSE;
} else {
MSG("b1 != 0")
(*ppr_p)->tree_b = 0;
p1->tree_b = 0;
}
*ppr_p = p1;
} else {
MSG("double LR")
p2 = p1->tree_r;
b2 = p2->tree_b;
p1->tree_r = p2->tree_l;
p2->tree_l = p1;
(*ppr_p)->tree_l = p2->tree_r;
p2->tree_r = *ppr_p;
if (b2 == -1)
(*ppr_p)->tree_b = 1;
else
(*ppr_p)->tree_b = 0;
if (b2 == 1)
p1->tree_b = -1;
else
p1->tree_b = 0;
*ppr_p = p2;
p2->tree_b = 0;
}
}
EXITV
}
int tree_trav(ppr_tree, pfi_uar)
tree **ppr_tree;
int (*pfi_uar)();
{
ENTER("tree_trav")
if (!*ppr_tree)
EXIT(TRUE)
if (!tree_trav(&(**ppr_tree).tree_l, pfi_uar))
EXIT(FALSE)
if (!(*pfi_uar)((**ppr_tree).tree_p))
EXIT(FALSE)
if (!tree_trav(&(**ppr_tree).tree_r, pfi_uar))
EXIT(FALSE)
EXIT(TRUE)
}
void tree_mung(ppr_tree, pfi_uar)
tree **ppr_tree;
int (*pfi_uar)();
{
ENTER("tree_mung")
if (*ppr_tree)
{
tree_mung(&(**ppr_tree).tree_l, pfi_uar);
tree_mung(&(**ppr_tree).tree_r, pfi_uar);
if (pfi_uar)
(*pfi_uar)((**ppr_tree).tree_p);
free(*ppr_tree);
*ppr_tree = NULL;
}
EXITV
}