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
|
package bjc.utils.funcdata.bst;
import java.util.ArrayList;
import java.util.Comparator;
import java.util.List;
import java.util.function.Predicate;
import bjc.utils.funcdata.FunctionalList;
import bjc.utils.funcdata.IList;
/**
* A binary search tree, with some mild support for functional traversal.
*
* @author ben
*
* @param <T>
* The data type stored in the node.
*/
public class BinarySearchTree<T> {
/*
* The comparator for use in ordering items
*/
private Comparator<T> comparator;
/*
* The current count of elements in the tree
*/
private int elementCount;
/*
* The root element of the tree
*/
private ITreePart<T> root;
/**
* Create a new tree using the specified way to compare elements.
*
* @param cmp
* The thing to use for comparing elements
*/
public BinarySearchTree(Comparator<T> cmp) {
if (cmp == null) {
throw new NullPointerException("Comparator must not be null");
}
elementCount = 0;
comparator = cmp;
}
/**
* Add a node to the binary search tree.
*
* @param element
* The data to add to the binary search tree.
*/
public void addNode(T element) {
elementCount++;
if (root == null) {
root = new BinarySearchTreeNode<>(element, null, null);
} else {
root.add(element, comparator);
}
}
/**
* Check if an adjusted pivot falls with the bounds of a list
*
* @param elements
* The list to get bounds from
* @param pivot
* The pivot
* @param pivotAdjustment
* The distance from the pivot
* @return Whether the adjusted pivot is with the list
*/
private boolean adjustedPivotInBounds(IList<T> elements, int pivot, int pivotAdjustment) {
return (pivot - pivotAdjustment) >= 0 && (pivot + pivotAdjustment) < elements.getSize();
}
/**
* Balance the tree, and remove soft-deleted nodes for free.
*
* Takes O(N) time, but also O(N) space.
*/
public void balance() {
IList<T> elements = new FunctionalList<>();
// Add each element to the list in sorted order
root.forEach(TreeLinearizationMethod.INORDER, element -> elements.add(element));
// Clear the tree
root = null;
// Set up the pivot and adjustment for readding elements
int pivot = elements.getSize() / 2;
int pivotAdjustment = 0;
// Add elements until there aren't any left
while (adjustedPivotInBounds(elements, pivot, pivotAdjustment)) {
if (root == null) {
// Create a new root element
root = new BinarySearchTreeNode<>(elements.getByIndex(pivot), null, null);
} else {
// Add the left and right elements in a balanced manner
root.add(elements.getByIndex(pivot + pivotAdjustment), comparator);
root.add(elements.getByIndex(pivot - pivotAdjustment), comparator);
}
// Increase the distance from the pivot
pivotAdjustment++;
}
// Add any trailing unbalanced elements
if ((pivot - pivotAdjustment) >= 0) {
root.add(elements.getByIndex(pivot - pivotAdjustment), comparator);
} else if ((pivot + pivotAdjustment) < elements.getSize()) {
root.add(elements.getByIndex(pivot + pivotAdjustment), comparator);
}
}
/**
* Soft-delete a node from the tree.
*
* Soft-deleted nodes stay in the tree until trim()/balance() is invoked, and
* are not included in traversals/finds.
*
* @param element
* The node to delete
*/
public void deleteNode(T element) {
elementCount--;
root.delete(element, comparator);
}
/**
* Get the root of the tree.
*
* @return The root of the tree.
*/
public ITreePart<T> getRoot() {
return root;
}
/**
* Check if a node is in the tree
*
* @param element
* The node to check the presence of for the tree.
* @return Whether or not the node is in the tree.
*/
public boolean isInTree(T element) {
return root.contains(element, comparator);
}
/**
* Traverse the tree in a specified way until the function fails
*
* @param linearizationMethod
* The way to linearize the tree for traversal
* @param traversalPredicate
* The function to use until it fails
*/
public void traverse(TreeLinearizationMethod linearizationMethod, Predicate<T> traversalPredicate) {
if (linearizationMethod == null) {
throw new NullPointerException("Linearization method must not be null");
} else if (traversalPredicate == null) {
throw new NullPointerException("Predicate must not be nulls");
}
root.forEach(linearizationMethod, traversalPredicate);
}
/**
* Remove all soft-deleted nodes from the tree.
*/
public void trim() {
List<T> nodes = new ArrayList<>(elementCount);
// Add all non-soft deleted nodes to the tree in insertion order
traverse(TreeLinearizationMethod.PREORDER, node -> {
nodes.add(node);
return true;
});
// Clear the tree
root = null;
// Add the nodes to the tree in the order they were inserted
nodes.forEach(node -> addNode(node));
}
}
|