-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathbst.cpp
161 lines (133 loc) · 4.27 KB
/
bst.cpp
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
#include <iostream>
#include <cassert>
template<class T>
class BinarySearchTree {
private:
struct Node {
Node(T data) : left{}, right{}, datum{data} { }
Node *left;
Node *right;
T datum;
};
public:
using size_type = size_t;
BinarySearchTree() : root{}, sz{0} { }
// Inserts the given data into the tree.
// MODIFIES: this
void insert(const T &data) {
insert_node(new Node(data), root);
++sz;
}
// Returns whether or not a given data value exists in the tree
bool exists(const T &data) const {
return exists_helper(data, root);
}
// Removes the first node with the given data
// MODIFIES: this
void remove(const T &data) {
assert(sz > 0);
remove_helper(data, root);
--sz;
}
// Returns the size of the tree
size_type size() const {
return sz;
}
// Does a pre-order print of the tree
void pre_order_print() const {
std::cout << "Pre-order Tree:\n";
pre_order(root, [] (const T &data) {
std::cout << data << " ";
});
std::cout << "\n";
}
void in_order_print() const {
std::cout << "In-order Tree:\n";
in_order(root, [] (const T &data) {
std::cout << data << " ";
});
std::cout << "\n";
}
private:
Node *root;
size_type sz;
// Recursively inserts a given node starting at the given root
static void insert_node(Node *n, Node *&root) {
if (!root)
root = n;
else if (n->datum < root->datum)
insert_node(n, root->left);
else
insert_node(n, root->right);
}
// Recursively searches the tree, starting at the given root,
// for the given item. Returns true if found, false otherwise.
static bool exists_helper(const T &data, const Node *root) {
if (!root)
return false;
else if (data < root->datum)
return exists_helper(data, root->left);
else if (data > root->datum)
return exists_helper(data, root->right);
else
return true;
}
// Removes the first node with given data, rooted at root
static void remove_helper(const T &data, Node *&root) {
if (!root) return;
if (data < root->datum)
remove_helper(data, root->left);
else if (data > root->datum)
remove_helper(data, root->right);
else if (data == root->datum) {
if (!root->left) {
root = root->right;
delete root;
} else if (!root->right) {
root = root->left;
delete root;
} else {
// Use inorder successor since the node were
// deleting has two children
const Node *successor = inorder_successor(root);
// Swap successor with node were deleting
root->datum = successor->datum;
// Delete 'successor' duplicate since we swapped it
remove_helper(successor->datum, root->right);
}
}
}
// Returns the inorder successor rooted at the given node
static const Node * inorder_successor(const Node *root) {
if (!root) return nullptr;
if (!root->right) return nullptr;
root = root->right;
while (root->left)
root = root->left;
return root;
}
// Does a pre-order traversal and performs a given action on nodes
template<typename Action>
static void pre_order(const Node *node, const Action &action) {
if (!node) return;
action(node->datum);
pre_order(node->left, action);
pre_order(node->right, action);
}
// Does in-order traversal and performs given action on nodes
template<typename Action>
static void in_order(const Node *node, const Action &action) {
if (!node) return;
in_order(node->left, action);
action(node->datum);
in_order(node->right, action);
}
// Does a post-order traversal and performs action on nodes
template<typename Action>
static void post_order(const Node *node, const Action &action) {
if (!node) return;
post_order(node->left, action);
post_order(node->right, action);
action(node->datum);
}
};