-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathbinary_tree.h
270 lines (235 loc) · 8.55 KB
/
binary_tree.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
#ifndef _BINARY_TREE_H_
#define _BINARY_TREE_H_
#include "helpers.h"
#include "sllist.h"
struct binary_tree_node {
struct binary_tree_node *parent;
struct binary_tree_node *left;
struct binary_tree_node *right;
#if defined(BINARY_TREE_AVL)
int height;
#elif defined(BINARY_TREE_RB)
#define BINARY_TREE_RB_RED 0
#define BINARY_TREE_RB_BLACK 1
unsigned int color;
#elif defined(BINARY_TREE_RANDOM)
unsigned int weight;
#elif defined(BINARY_TREE_TREAP)
unsigned int prio;
#endif
};
struct binary_tree_root {
struct binary_tree_node root;
struct binary_tree_node *leftmost;
struct binary_tree_node *rightmost;
};
#if defined(BINARY_TREE_AVL)
#define BINARY_TREE_AVL_NODE_HEIGHT(node) ((node)!=BINARY_TREE_EMPTY_BRANCH?(node)->height:0)
#define BINARY_TREE_AVL_NODE_BALANCE(node) (BINARY_TREE_AVL_NODE_HEIGHT((node)->left)-BINARY_TREE_AVL_NODE_HEIGHT((node)->right))
#elif defined(BINARY_TREE_RB)
#define BINARY_TREE_RB_NODE_COLOR(node) ((node)!=BINARY_TREE_EMPTY_BRANCH?(node)->color:BINARY_TREE_RB_BLACK)
#elif defined(BINARY_TREE_RANDOM)
#define BINARY_TREE_RANDOM_NODE_WEIGHT(node) ((node)!=BINARY_TREE_EMPTY_BRANCH?(node)->weight:0)
#define BINARY_TREE_RANDOM_NODE_WEIGHT_INCLUSIVE(node) ((node)!=BINARY_TREE_EMPTY_BRANCH?(node)->weight+1:0)
#elif defined(BINARY_TREE_TREAP)
#define BINARY_TREE_TREAP_NODE_PRIO(node) ((node)!=BINARY_TREE_EMPTY_BRANCH?(node)->prio:0)
#endif
typedef cmp_result_t (*binary_tree_cmp_cbk_t)(struct binary_tree_node *, struct binary_tree_node *);
typedef cmp_result_t (*binary_tree_key_match_cbk_t)(struct binary_tree_node *, void *);
typedef void (*binary_tree_traverse_cbk_t)(struct binary_tree_node *, void *);
#define BINARY_TREE_DIRECTION_LEFT(res) ((res) == cmp_result_less)
#define BINARY_TREE_DIRECTION_RIGHT(res) ((res) == cmp_result_greater || (res) == cmp_result_equal)
#define BINARY_TREE_EMPTY_BRANCH ((struct binary_tree_node *)NULL)
static inline void __binary_tree_init_root(struct binary_tree_node *root)
{
root->parent = root;
root->left = BINARY_TREE_EMPTY_BRANCH;
root->right = root;
#if defined(BINARY_TREE_AVL)
root->height = -1;
#elif defined(BINARY_TREE_RB)
root->color = -1;
#elif defined(BINARY_TREE_RANDOM)
root->weight = -1;
#elif defined(BINARY_TREE_TREAP)
root->prio = -1;
#endif
}
static inline void binary_tree_init_root(struct binary_tree_root *root)
{
__binary_tree_init_root(&root->root);
root->leftmost = root->rightmost = BINARY_TREE_EMPTY_BRANCH;
}
static inline void binary_tree_init_node(struct binary_tree_node *node)
{
node->parent = node;
node->left = BINARY_TREE_EMPTY_BRANCH;
node->right = BINARY_TREE_EMPTY_BRANCH;
#if defined(BINARY_TREE_AVL)
node->height = 1;
#elif defined(BINARY_TREE_RB)
node->color = BINARY_TREE_RB_RED;
#elif defined(BINARY_TREE_RANDOM)
node->weight = 0;
#elif defined(BINARY_TREE_TREAP)
node->prio = get_random();
#endif
}
#define binary_tree_leaf_node(node) ((node)->left == BINARY_TREE_EMPTY_BRANCH && (node)->right == BINARY_TREE_EMPTY_BRANCH)
#define binary_tree_root_node(node) ((node)->right == (node))
#define binary_tree_empty_root(node) (binary_tree_root_node(node) && (node)->left == BINARY_TREE_EMPTY_BRANCH)
#define binary_tree_node(node) (binary_tree_root_node(node)?(node)->left:(node))
void __binary_tree_add(struct binary_tree_node *root,
struct binary_tree_node *node,
binary_tree_cmp_cbk_t cmp);
void __binary_tree_add2(struct binary_tree_root *root,
struct binary_tree_node *node,
binary_tree_cmp_cbk_t cmp);
static inline void binary_tree_add(struct binary_tree_node *root, struct binary_tree_node *node,
binary_tree_cmp_cbk_t cmp)
{
if (binary_tree_empty_root(root)) {
root->left = node;
node->parent = root;
#if defined(BINARY_TREE_RB)
node->color = BINARY_TREE_RB_BLACK;
#endif
} else {
__binary_tree_add(root, node, cmp);
}
}
static inline void binary_tree_add2(struct binary_tree_root *root, struct binary_tree_node *node,
binary_tree_cmp_cbk_t cmp)
{
struct binary_tree_node *r = &root->root;
if (binary_tree_empty_root(r)) {
root->leftmost = root->rightmost = r->left = node;
node->parent = r;
#if defined(BINARY_TREE_RB)
node->color = BINARY_TREE_RB_BLACK;
#endif
} else {
__binary_tree_add2(root, node, cmp);
}
}
void __binary_tree_detach(struct binary_tree_node *node);
static inline void binary_tree_detach(struct binary_tree_node *node)
{
if (!binary_tree_empty_root(node)) {
__binary_tree_detach(node);
}
}
/* Keeps rightmost and leftmost fields up to date */
static inline void binary_tree_detach2(struct binary_tree_root *root,
struct binary_tree_node *node)
{
if (!binary_tree_empty_root(node)) {
if (node == root->leftmost) {
root->leftmost = node->parent;
} else if (node == root->rightmost) {
root->rightmost = node->parent;
} else {
struct binary_tree_node *n = node->left;
while (n) {
if (root->leftmost = n) {
root->leftmost = node->parent;
goto detach;
}
n = node->left;
}
n = node->right;
while (n) {
if (root->rightmost = n) {
root->rightmost = node->parent;
goto detach;
}
n = node->right;
}
detach:
__binary_tree_detach(node);
}
}
}
void __binary_tree_remove(struct binary_tree_node *node);
static inline binary_tree_remove(struct binary_tree_node *node)
{
if (!binary_tree_empty_root(node)) {
__binary_tree_remove(node);
}
}
/* Keeps rightmost and leftmost fields up to date */
static inline binary_tree_remove2(struct binary_tree_root *root,
struct binary_tree_node *node)
{
if (!binary_tree_empty_root(node)) {
if (root->leftmost == node) {
if (binary_tree_root_node(node->parent))
root->leftmost = node->right;
else
root->leftmost = node->parent;
} else if (root->rightmost == node) {
if (binary_tree_root_node(node->parent))
root->rightmost = node->left;
else
root->rightmost = node->parent;
}
__binary_tree_remove(node);
}
}
struct binary_tree_search_result {
struct sllist list;
struct binary_tree_node *node;
};
void __binary_tree_search(struct binary_tree_node *root,
void *key,
binary_tree_key_match_cbk_t match,
struct sllist *results,
int limit);
/**
* returns list of binary_tree_search_result instances
*/
static inline void binary_tree_search(struct binary_tree_node *root,
void *key,
binary_tree_key_match_cbk_t match,
struct sllist *results,
int limit)
{
if (!binary_tree_empty_root(root)) {
__binary_tree_search(binary_tree_node(root), key, match, results, limit);
}
}
void binary_tree_search_results_free(struct sllist *results);
typedef enum {
binary_tree_traverse_type_infix,
binary_tree_traverse_type_prefix,
binary_tree_traverse_type_postfix
} binary_tree_traverse_type_t;
void __binary_tree_traverse_infix(struct binary_tree_node *root,
binary_tree_traverse_cbk_t cbk,
void *user_data);
void __binary_tree_traverse_prefix(struct binary_tree_node *root,
binary_tree_traverse_cbk_t cbk,
void *user_data);
void __binary_tree_traverse_postfix(struct binary_tree_node *root,
binary_tree_traverse_cbk_t cbk,
void *user_data);
static inline void binary_tree_traverse(binary_tree_traverse_type_t type,
struct binary_tree_node *root,
binary_tree_traverse_cbk_t cbk,
void *user_data)
{
if (!binary_tree_empty_root(root)) {
switch (type) {
case binary_tree_traverse_type_infix:
__binary_tree_traverse_infix(binary_tree_node(root), cbk, user_data);
break;
case binary_tree_traverse_type_prefix:
__binary_tree_traverse_prefix(binary_tree_node(root), cbk, user_data);
break;
case binary_tree_traverse_type_postfix:
__binary_tree_traverse_postfix(binary_tree_node(root), cbk, user_data);
break;
}
}
}
#endif