forked from Suryakant-Bharti/Important-Java-Concepts
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathBinarySearchTree.kt
170 lines (141 loc) Β· 4.27 KB
/
BinarySearchTree.kt
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
package algo.datastructures.tree
import algo.datastructures.Queue
import java.util.NoSuchElementException
class BinarySearchTree<K: Comparable<K>, V>: Map<K, V> {
data class Node<K, V>(
override val key: K,
override var value: V,
var left: Node<K, V>? = null,
var right: Node<K, V>? = null,
var size: Int = 1): Map.Entry<K, V>
private var root: Node<K, V>? = null
override val size: Int
get() = size(root)
override val entries: Set<Map.Entry<K, V>>
get() {
val set = mutableSetOf<Node<K, V>>()
inorder(root) { set.add(it.copy()) }
return set
}
override val keys: Set<K>
get() {
val set = mutableSetOf<K>()
inorder(root) { set.add(it.key) }
return set
}
override val values: Collection<V>
get() {
val queue = Queue<V>()
inorder(root) { queue.add(it.value) }
return queue
}
override fun get(key: K): V? {
var x = root
while (x != null) {
x = when {
key < x.key -> x.left
key > x.key -> x.right
else -> return x.value
}
}
return null
}
override fun containsKey(key: K): Boolean {
return get(key) != null
}
override fun containsValue(value: V): Boolean {
return any { it.value == value }
}
fun add(key: K, value: V) {
root = add(key, value, root)
}
private fun add(key: K, value: V, x: Node<K, V>?): Node<K, V> {
if (x == null) return Node(key, value)
when {
key < x.key -> x.left = add(key, value, x.left)
key > x.key -> x.right = add(key, value, x.right)
else -> x.value = value
}
x.size = size(x.left) + size(x.right) + 1
return x
}
fun remove(key: K) {
root = remove(key, root)
}
private fun remove(key: K, root: Node<K, V>?): Node<K, V>? {
var x: Node<K, V> = root ?: throw NoSuchElementException()
when {
key < x.key -> x.left = remove(key, x.left)
key > x.key -> x.right = remove(key, x.right)
else -> {
if (x.left == null) return x.right
if (x.right == null) return x.left
val tmp = x
x = pollMin(tmp.right!!)!!
x.right = min(tmp.right)
x.left = tmp.left
}
}
x.size = size(x.left) + size(x.right) + 1
return x
}
private fun size(x: Node<K, V>?): Int = x?.size ?: 0
fun height(): Int {
return height(root)
}
private fun height(x: Node<K, V>?): Int {
if (x == null) return 0
return maxOf(height(x.left), height(x.right)) + 1
}
override fun isEmpty(): Boolean {
return size == 0
}
fun min(): K {
return min(root).key
}
fun min(node: Node<K, V>?): Node<K, V> {
if (node == null) throw NoSuchElementException()
var x: Node<K, V> = node
while (x.left != null) {
x = x.left!!
}
return x
}
fun max(): K {
return max(root).key
}
fun max(node: Node<K, V>?): Node<K, V> {
if (node == null) throw NoSuchElementException()
var x: Node<K, V> = node
while (x.right != null) {
x = x.right!!
}
return x
}
fun pollMin() {
if (root == null) throw NoSuchElementException()
root = pollMin(root!!)
}
private fun pollMin(x: Node<K, V>): Node<K, V>? {
if (x.left == null) return x.right
x.left = pollMin(x.left!!)
x.size = size(x.left) + size(x.right) + 1
return x
}
fun pollMax() {
if (root == null) throw NoSuchElementException()
root = pollMax(root!!)
}
private fun pollMax(x: Node<K, V>): Node<K, V>? {
if (x.right == null) return x.left
x.right = pollMax(x.right!!)
x.size = size(x.left) + size(x.right) + 1
return x
}
private fun inorder(x: Node<K, V>?, lambda: (Node<K, V>) -> (Unit)) {
if (x == null) return
inorder(x.left, lambda)
lambda(x)
inorder(x.right, lambda)
}
}