245 lines
5.9 KiB
C++
245 lines
5.9 KiB
C++
/* $Header$ */
|
|
|
|
#include <alloc.h>
|
|
|
|
/* Implementation of AVL-trees: trees in which the difference in depth
|
|
of the left branch and the right branch is at most one.
|
|
The difference in depth is indicated by a "balance" flag in each node:
|
|
this flag has one of three values:
|
|
. indicating that the left branch has the same depth as the right branch,
|
|
+ indicating that the right branch is deeper,
|
|
- indicating that the left branch is deeper.
|
|
So, a node has the following structure:
|
|
*/
|
|
|
|
struct avl_node {
|
|
struct avl_node
|
|
*left,
|
|
*right; /* the left and right branches */
|
|
char *info; /* pointer to information in this node */
|
|
char balance; /* balance information described above */
|
|
};
|
|
|
|
/* create definitions for new_avl_node() and free_avl_node() */
|
|
/* STATICALLOCDEF "avl_node" 10 */
|
|
|
|
/* There is also a tree header, which contains the root of the tree and
|
|
the address of a user-supplied comparison routine:
|
|
*/
|
|
|
|
struct avl_tree {
|
|
struct avl_node
|
|
*root; /* root of the avl tree */
|
|
int (*cmp)(); /* address of comparison routine */
|
|
};
|
|
/* create definitions for new_avl_tree() and free_avl_tree() */
|
|
/* STATICALLOCDEF "avl_tree" 2 */
|
|
|
|
/* The next routine adds a node to an avl tree. It returns 1 if the
|
|
tree got deeper.
|
|
*/
|
|
static int
|
|
balance_add(ppsc, n, cmp)
|
|
struct avl_node **ppsc; /* address of root */
|
|
register char *n; /* user-supplied information */
|
|
int (*cmp)(); /* user-supplied comparison routine */
|
|
{
|
|
register struct avl_node *psc = *ppsc, *qsc, *ssc;
|
|
|
|
if (! psc) {
|
|
*ppsc = new_avl_node();
|
|
(*ppsc)->balance = '.';
|
|
(*ppsc)->info = n;
|
|
return 1;
|
|
}
|
|
if ((*cmp)(n, psc->info) < 0) {
|
|
if (balance_add(&(psc->left), n, cmp)) {
|
|
/* left hand side got deeper */
|
|
if (psc->balance == '+') {
|
|
/* but the right hand side was deeper */
|
|
psc->balance = '.';
|
|
return 0;
|
|
}
|
|
if (psc->balance == '.') {
|
|
/* but the right hand side was as deep */
|
|
psc->balance = '-';
|
|
return 1;
|
|
}
|
|
/* left hand side already was one deeper; re-organize */
|
|
qsc = psc->left;
|
|
if (qsc->balance == '-') {
|
|
/* if left-hand side of left node was deeper,
|
|
this node becomes the new root
|
|
*/
|
|
psc->balance = '.';
|
|
qsc->balance = '.';
|
|
psc->left = qsc->right;
|
|
qsc->right = psc;
|
|
*ppsc = qsc;
|
|
return 0;
|
|
}
|
|
/* else the right node of the left node becomes the new root */
|
|
ssc = qsc->right;
|
|
psc->left = ssc->right;
|
|
qsc->right = ssc->left;
|
|
ssc->left = qsc;
|
|
ssc->right = psc;
|
|
*ppsc = ssc;
|
|
if (ssc->balance == '.') {
|
|
psc->balance = '.';
|
|
qsc->balance = '.';
|
|
return 0;
|
|
}
|
|
if (ssc->balance == '-') {
|
|
psc->balance = '+';
|
|
qsc->balance = '.';
|
|
ssc->balance = '.';
|
|
return 0;
|
|
}
|
|
psc->balance = '.';
|
|
qsc->balance = '-';
|
|
}
|
|
return 0;
|
|
}
|
|
if (balance_add(&(psc->right), n, cmp)) {
|
|
/* right hand side got deeper */
|
|
if (psc->balance == '-') {
|
|
/* but the left hand side was deeper */
|
|
psc->balance = '.';
|
|
return 0;
|
|
}
|
|
if (psc->balance == '.') {
|
|
/* but the left hand side as deep */
|
|
psc->balance = '+';
|
|
return 1;
|
|
}
|
|
/* right hand side already was one deeper; re-organize */
|
|
qsc = psc->right;
|
|
if (qsc->balance == '+') {
|
|
/* if right-hand side of left node was deeper,
|
|
this node becomes the new root
|
|
*/
|
|
psc->balance = '.';
|
|
qsc->balance = '.';
|
|
psc->right = qsc->left;
|
|
qsc->left = psc;
|
|
*ppsc = qsc;
|
|
return 0;
|
|
}
|
|
/* else the left node of the right node becomes the new root */
|
|
ssc = qsc->left;
|
|
psc->right = ssc->left;
|
|
qsc->left = ssc->right;
|
|
ssc->right = qsc;
|
|
ssc->left = psc;
|
|
*ppsc = ssc;
|
|
if (ssc->balance == '.') {
|
|
psc->balance = '.';
|
|
qsc->balance = '.';
|
|
return 0;
|
|
}
|
|
if (ssc->balance == '+') {
|
|
psc->balance = '-';
|
|
qsc->balance = '.';
|
|
ssc->balance = '.';
|
|
return 0;
|
|
}
|
|
psc->balance = '.';
|
|
qsc->balance = '+';
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* extern struct avl_tree *create_avl_tree(int (*cmp)());
|
|
Returns a fresh avl_tree structure.
|
|
*/
|
|
struct avl_tree *
|
|
create_avl_tree(cmp)
|
|
int (*cmp)(); /* comparison routine */
|
|
{
|
|
register struct avl_tree *p = new_avl_tree();
|
|
|
|
p->cmp = cmp;
|
|
return p;
|
|
}
|
|
|
|
/* extern add_to_avl_tree(struct avl_tree *tree, char *n);
|
|
Adds the information indicated by 'n' to the avl_tree indicated by 'tree'
|
|
*/
|
|
add_to_avl_tree(tree, n)
|
|
struct avl_tree *tree; /* tree to be added to */
|
|
char *n; /* information */
|
|
{
|
|
(void) balance_add(&(tree->root), n, tree->cmp);
|
|
}
|
|
|
|
/* extern char *find_ngt(struct avl_tree *tree, char *n);
|
|
Returns the information in the largest node that still compares <= to 'n',
|
|
or 0 if not present.
|
|
*/
|
|
char *
|
|
find_ngt(tree, n)
|
|
struct avl_tree *tree; /* tree to be searched in */
|
|
char *n; /* information to be compared with */
|
|
{
|
|
register struct avl_node *nd = tree->root, *lastnd = 0;
|
|
|
|
for (;;) {
|
|
while (nd && (*tree->cmp)(nd->info, n) > 0) {
|
|
nd = nd->left;
|
|
}
|
|
while (nd && (*tree->cmp)(nd->info, n) <= 0) {
|
|
lastnd = nd;
|
|
nd = nd->right;
|
|
}
|
|
if (! nd) break;
|
|
}
|
|
return lastnd ? lastnd->info : (char *) 0;
|
|
}
|
|
|
|
/* extern char *find_nlt(struct avl_tree *tree, char *n);
|
|
Returns the information in the largest node that still compares >= to 'n',
|
|
or 0 if not present.
|
|
*/
|
|
char *
|
|
find_nlt(tree, n)
|
|
struct avl_tree *tree; /* tree to be searched in */
|
|
char *n; /* information to be compared with */
|
|
{
|
|
register struct avl_node *nd = tree->root, *lastnd = 0;
|
|
|
|
for (;;) {
|
|
while (nd && (*tree->cmp)(nd->info, n) < 0) {
|
|
nd = nd->right;
|
|
}
|
|
while (nd && (*tree->cmp)(nd->info, n) >= 0) {
|
|
lastnd = nd;
|
|
nd = nd->left;
|
|
}
|
|
if (! nd) break;
|
|
}
|
|
return lastnd ? lastnd->info : (char *) 0;
|
|
}
|
|
|
|
/* extern char *find_eq(struct avl_tree *tree, char *n);
|
|
Returns the information in the node that compares equal to 'n',
|
|
or 0 if not present.
|
|
*/
|
|
char *
|
|
find_eq(tree, n)
|
|
struct avl_tree *tree; /* tree to be searched in */
|
|
char *n; /* information to be compared with */
|
|
{
|
|
register struct avl_node *nd = tree->root;
|
|
|
|
for (;;) {
|
|
while (nd && (*tree->cmp)(nd->info, n) < 0) {
|
|
nd = nd->right;
|
|
}
|
|
while (nd && (*tree->cmp)(nd->info, n) > 0) {
|
|
nd = nd->left;
|
|
}
|
|
if (! nd) break;
|
|
}
|
|
return nd ? nd->info : (char *) 0;
|
|
}
|