-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.ts
182 lines (162 loc) · 4.97 KB
/
index.ts
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
export class Node<Number> {
value: Number
left: Node<Number> | null
right: Node<Number> | null
constructor(value: Number) {
this.value = value
this.left = null
this.right = null
}
/**
* Finds the minimum value in the binary tree
* @returns {Node<Number>} The node containing the minimum value in the binary tree
*/
findMin(): Node<Number> {
// we know that the minimum value in a binary search tree is the leftmost node
// so we recurse to the left until we find a node that has no left child
if (!this.left) {
return this
} else {
return this.left.findMin()
}
}
}
export default class BinaryTree<Number> {
_root: Node<Number> | null
private insertRecursion = (
node: Node<Number> | null,
value: Number
): Node<Number> => {
if (node === null) {
return new Node(value)
}
if (value <= node.value) {
// if the left child is not null, we recurse to the left
node.left = this.insertRecursion(node.left, value)
} else if (value > node.value) {
// if the right child is not null, we recurse to the right
node.right = this.insertRecursion(node.right, value)
} else {
// if the value is equal to the current node's value, we don't insert it
return node
}
return node
}
private findRecursion = (
node: Node<Number> | null,
value: Number
): Node<Number> | null => {
if (!node) {
return null
} else if (node.value === value) {
// we found the node with the value we are looking for
return node
} else if (value < node.value) {
// traverse left
return this.findRecursion(node.left, value)
}
// traverse right
return this.findRecursion(node.right, value)
}
private removeRecursion = (
node: Node<Number> | null,
value: Number
): Node<Number> | null => {
if (!node) {
return null
} else if (value < node.value) {
// if the value we want to remove is less than the current node's value,
// we recurse to the left
node.left = this.removeRecursion(node.left, value)
} else if (value > node.value) {
// if the value we want to remove is greater than the current node's value,
// we recurse to the right
node.right = this.removeRecursion(node.right, value)
} else {
if (node.left === null && node.right === null) {
// if the node has no children, we can just remove it
node = null
} else if (node.left == null) {
// if the node has only a right child, we can replace the node with its right child
node = node.right
} else if (node.right == null) {
// if the node has only a left child, we can replace the node with its left child
node = node.left
} else {
// if the node has two children, we find the minimum value in the right subtree
const minRight = node.right.findMin()
node.value = minRight.value
node.right = this.removeRecursion(node.right, minRight.value)
}
}
return node
}
/**
* Returns the root node of the binary tree
* @returns {Node<Number> | null} The root node of the binary tree
*/
get root(): Node<Number> | null {
return this._root
}
/**
* Creates a new binary tree
* @param {Number} initialValue The initial value of the binary tree
*/
constructor(initialValue?: Number) {
this._root = initialValue ? new Node(initialValue) : null
}
/**
* Clears the binary tree
*/
public clear(): void {
this._root = null
}
/**
* Inserts a value into the binary tree
* @param {Number} value The value to insert into the binary tree
*/
public insert(value: Number): void {
if (!this._root) {
this._root = new Node(value)
return
}
this.insertRecursion(this._root, value)
}
/**
* Finds a value in the binary tree
* @param {Number} value The value to find in the binary tree
* @returns {Node<Number> | null} The node containing the value or null if the value is not found
*/
public find(value: Number): Node<Number> | null {
return this.findRecursion(this._root, value)
}
/**
* Removes a value from the binary tree
* @param {Number} value The value to remove from the binary tree
*/
public remove(value: Number): void {
this.removeRecursion(this._root, value)
}
/**
* Prints the binary tree in a console-friendly format that clearly illustrates the tree structure.
* @returns {void}
*/
public print(): void {
const printNode = (
node: Node<Number> | null,
prefix: string = '',
isLeft: boolean = true
) => {
if (node !== null) {
console.log(prefix + (isLeft ? '├── ' : '└── ') + node.value)
printNode(node.left, prefix + (isLeft ? '│ ' : ' '), true)
printNode(node.right, prefix + (isLeft ? '│ ' : ' '), false)
}
}
if (this._root === null) {
console.log('Tree is empty')
} else {
printNode(this._root, '', true)
}
}
}