Commit 07cb206931c6a135a0d038c7a6a06ceb5507d427
1 parent
af8994b0
Created a macro for tree find to prevent duplicating this code in insert and delete.
Showing
5 changed files
with
53 additions
and
80 deletions
... | ... | @@ -29,6 +29,27 @@ |
29 | 29 | #define TR_TREE_LEFT(node) (NULL!=(node)?(node)->left:NULL) |
30 | 30 | #define TR_TREE_PARENT(node) (NULL!=(node)?(node)->parent:NULL) |
31 | 31 | |
32 | +/* | |
33 | + * Find data in a tree. | |
34 | + * Attention: This will change node, so normally you need to copy | |
35 | + * it before using this macro. | |
36 | + * Also be aware that found needs to be a valid lvalue and an integer. | |
37 | + */ | |
38 | +#define TR_TREE_FIND(node, search, found, comp) \ | |
39 | + (found) = -1; \ | |
40 | + while((node) && ((node)->left || (node)->right)) { \ | |
41 | + (found) = (comp)((node)->data, (search)); \ | |
42 | + if (0 != (found)) { \ | |
43 | + if (0 < (found)) { \ | |
44 | + (node) = (node)->left; \ | |
45 | + } else { \ | |
46 | + (node) = (node)->right; \ | |
47 | + } \ | |
48 | + } else { \ | |
49 | + break; \ | |
50 | + } \ | |
51 | + } | |
52 | + | |
32 | 53 | #define TR_TREE_CHILD(node) \ |
33 | 54 | (NULL==TR_TREE_RIGHT((node))?TR_TREE_LEFT((node)):TR_TREE_RIGHT((node))) |
34 | 55 | ... | ... |
... | ... | @@ -28,35 +28,16 @@ TR_treeDelete(TR_Tree * this, const void * search, TR_TreeComp comp) |
28 | 28 | { |
29 | 29 | TR_Tree node = *this; |
30 | 30 | TR_Tree del_node; |
31 | + int found; | |
31 | 32 | |
32 | 33 | void * data; |
33 | 34 | |
34 | - /* | |
35 | - * first search for it and if its found return the data | |
36 | - * and we are done... | |
37 | - */ | |
38 | - while (NULL != node) { | |
39 | - int comparison = comp(node->data, search); | |
40 | - | |
41 | - if (0 < comparison) { | |
42 | - node = TR_TREE_LEFT(node); | |
43 | - continue; | |
44 | - } | |
45 | - | |
46 | - if (0 > comparison) { | |
47 | - node = TR_TREE_RIGHT(node); | |
48 | - continue; | |
49 | - } | |
50 | - | |
51 | - if (0 == comparison) { | |
52 | - break; | |
53 | - } | |
54 | - } | |
35 | + TR_TREE_FIND(node, search, found, comp); | |
55 | 36 | |
56 | 37 | /* |
57 | 38 | * nothing was found...return NULL to indicate this. |
58 | 39 | */ |
59 | - if (NULL == node) { | |
40 | + if (found != 0) { | |
60 | 41 | return NULL; |
61 | 42 | } |
62 | 43 | ... | ... |
... | ... | @@ -25,25 +25,12 @@ |
25 | 25 | void * |
26 | 26 | TR_treeFind(TR_Tree this, const void * search, TR_TreeComp comp) |
27 | 27 | { |
28 | - while (NULL != this) { | |
29 | - int comparison = comp(this->data, search); | |
28 | + TR_Tree node = this; | |
29 | + int found; | |
30 | 30 | |
31 | - if (0 < comparison) { | |
32 | - this = TR_TREE_LEFT(this); | |
33 | - continue; | |
34 | - } | |
31 | + TR_TREE_FIND(node, search, found, comp); | |
35 | 32 | |
36 | - if (0 > comparison) { | |
37 | - this = TR_TREE_RIGHT(this); | |
38 | - continue; | |
39 | - } | |
40 | - | |
41 | - if (0 == comparison) { | |
42 | - break; | |
43 | - } | |
44 | - } | |
45 | - | |
46 | - return NULL != this ? this->data : NULL; | |
33 | + return found == 0 ? node->data : NULL; | |
47 | 34 | } |
48 | 35 | |
49 | 36 | // vim: set ts=4 sw=4: | ... | ... |
... | ... | @@ -28,6 +28,9 @@ TR_treeInsert(TR_Tree * this, const void * search, TR_TreeComp comp) |
28 | 28 | { |
29 | 29 | TR_Tree node = *this; |
30 | 30 | TR_Tree new_node = NULL; |
31 | + int found; | |
32 | + | |
33 | + TR_TREE_FIND(node, search, found, comp); | |
31 | 34 | |
32 | 35 | /* |
33 | 36 | * insert the node or return the one in tree if comparison |
... | ... | @@ -40,43 +43,24 @@ TR_treeInsert(TR_Tree * this, const void * search, TR_TreeComp comp) |
40 | 43 | */ |
41 | 44 | *this = node = new_node = TR_new(TR_Tree, search); |
42 | 45 | } else { |
43 | - /* | |
44 | - * first search for it and if its found return the data | |
45 | - * and we are done... | |
46 | - */ | |
47 | - while (NULL != node) { | |
48 | - int comparison = comp(node->data, search); | |
49 | - | |
50 | - if (0 < comparison) { | |
51 | - if (NULL != TR_TREE_LEFT(node)) { | |
52 | - node = TR_TREE_LEFT(node); | |
53 | - continue; | |
54 | - } else { | |
55 | - node->left = TR_new(TR_Tree, search); | |
56 | - node->left->parent = node; | |
57 | - node = new_node = node->left; | |
58 | - break; | |
59 | - } | |
60 | - } | |
61 | - | |
62 | - if (0 > comparison) { | |
63 | - if (NULL != TR_TREE_RIGHT(node)) { | |
64 | - node = TR_TREE_RIGHT(node); | |
65 | - continue; | |
66 | - } else { | |
67 | - node->right = TR_new(TR_Tree, search); | |
68 | - node->right->parent = node; | |
69 | - node = new_node = node->right; | |
70 | - break; | |
71 | - } | |
72 | - } | |
73 | - | |
74 | - if (0 == comparison) { | |
75 | - // as this is insert and not find this will overwrite an existing value, | |
76 | - // but return the previos one so that it can be freed if neccessary. | |
77 | - void * data = node->data; | |
78 | - node->data = (void *)search; | |
79 | - return data; | |
46 | + if (found == 0) { | |
47 | + // we found an element | |
48 | + // as this is insert and not find this will overwrite an existing | |
49 | + // value, but return the previous one so that it can be freed if | |
50 | + // neccessary. | |
51 | + void * data = node->data; | |
52 | + node->data = (void *)search; | |
53 | + return data; | |
54 | + } else { | |
55 | + // not found | |
56 | + if (0 < found) { | |
57 | + node->left = TR_new(TR_Tree, search); | |
58 | + node->left->parent = node; | |
59 | + node = new_node = node->left; | |
60 | + } else { | |
61 | + node->right = TR_new(TR_Tree, search); | |
62 | + node->right->parent = node; | |
63 | + node = new_node = node->right; | |
80 | 64 | } |
81 | 65 | } |
82 | 66 | } |
... | ... | @@ -114,13 +98,13 @@ TR_treeInsert(TR_Tree * this, const void * search, TR_TreeComp comp) |
114 | 98 | && node->parent == node->parent->parent->left) { |
115 | 99 | |
116 | 100 | TR_TREE_ROTATE(left, this, node->parent); |
117 | - node = TR_TREE_LEFT(node); | |
101 | + node = node->left; | |
118 | 102 | |
119 | 103 | } else if (node == node->parent->left |
120 | 104 | && node->parent == node->parent->parent->right) { |
121 | 105 | |
122 | 106 | TR_TREE_ROTATE(right, this, node->parent); |
123 | - node = TR_TREE_RIGHT(node); | |
107 | + node = node->right; | |
124 | 108 | |
125 | 109 | } |
126 | 110 | ... | ... |
... | ... | @@ -11,13 +11,13 @@ |
11 | 11 | #define HAVE_MEMORY_H 1 |
12 | 12 | |
13 | 13 | /* Define to 1 if you have the `memset' function. */ |
14 | -/* #undef HAVE_MEMSET */ | |
14 | +#define HAVE_MEMSET 1 | |
15 | 15 | |
16 | 16 | /* Define to 1 if you have the <stdarg.h> header file. */ |
17 | 17 | #define HAVE_STDARG_H 1 |
18 | 18 | |
19 | 19 | /* Define to 1 if stdbool.h conforms to C99. */ |
20 | -/* #undef HAVE_STDBOOL_H */ | |
20 | +#define HAVE_STDBOOL_H 1 | |
21 | 21 | |
22 | 22 | /* Define to 1 if you have the <stdint.h> header file. */ |
23 | 23 | #define HAVE_STDINT_H 1 | ... | ... |
Please
register
or
login
to post a comment