-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathscript.rb
329 lines (303 loc) · 9.1 KB
/
script.rb
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
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
# frozen_string_literal: true
class Node
def initialize(value = nil, left = nil, right = nil)
@value = value
@left = left
@right = right
end
attr_accessor :value, :left, :right
end
class Tree
def initialize(array = [])
@array = array
@root = nil
end
def root(root)
@root = root
end
def array(array)
@array = array
end
def build_tree(array = nil, start_arr = 0, end_arr = 0, count = 0)
return nil if start_arr > end_arr
array = @array.uniq.sort if count == 0 && array.nil?
count += 1
start_arr = 0
end_arr = array.length - 1
mid_arr = (start_arr + end_arr) / 2
root = Node.new(array[mid_arr])
root.left = build_tree(array[start_arr.. mid_arr-1], start_arr, mid_arr-1, count)
root.right = build_tree(array[mid_arr+1.. end_arr], mid_arr+1, end_arr, count)
root(root)
end
def pretty_print(node = @root, prefix = '', is_left = true)
pretty_print(node.right, "#{prefix}#{is_left ? '│ ' : ' '}", false) if node.right
puts "#{prefix}#{is_left ? '└── ' : '┌── '}#{node.value}"
pretty_print(node.left, "#{prefix}#{is_left ? ' ' : '│ '}", true) if node.left
end
def insert(value, root = nil, inserted = false)
root = @root if root == nil
return nil if inserted
inserted = false
if root.value > value
if root.left == nil
root.left = Node.new(value)
inserted = true
end
root = root.left
else
if root.right == nil
root.right = Node.new(value)
inserted = true
end
root = root.right
end
insert(value, root, inserted)
end
def delete(value, root = nil, deleted = false, not_found = false, current_root = nil)
return nil if deleted || not_found
return nil if !(@array.include?(value))
root = @root if root.nil?
if root.value < value
delete(value, root.right, deleted, not_found) if deleted == false
root.right = nil if root.right.value == value
return
elsif root.value > value
delete(value, root.left, deleted, not_found) if deleted == false
root.left = nil if root.left.value == value
return
elsif @root.value == value
current_root = @root.right if current_root.nil?
if !(current_root.left.nil?)
current_root = current_root.left
else
value = current_root.value
delete(current_root.value, @root, deleted, not_found)
@root.value = value
deleted = true
end
else
if root.value == value
if root.right.nil? && root.left.nil? # no child
root = nil
deleted = true
else
if root.value < value
delete(value, root.right, deleted, not_found) if deleted == false
elsif root.value > value
delete(value, root.left, deleted, not_found) if deleted == false
end
if root.left.nil? # 1 child
root.value = root.right.value
root.right = nil
deleted = true
elsif root.right.nil?
root.value = root.left.value
root.left = nil
deleted = true
end
if !(root.right.nil? && root.left.nil?) # 2 childs
current_root = root.right if current_root.nil?
if !(current_root.left.nil?)
current_root = current_root.left
else
value = current_root.value
delete(current_root.value, root, deleted, not_found)
root.value = value
deleted = true
end
end
end
end
end
not_found = true if root.nil?
delete(value, root, deleted, not_found, current_root)
end
def find(value, root = false, found = false)
root = @root if root == false
if !(root.nil?) && root.value == value
print "node: "
p root
found = true
end
return puts "#{!(root.nil?)}" if root.nil? || found == true
if root.value < value
find(value, root.right, found) if found == false
elsif root.value > value
find(value, root.left, found) if found == false
end
end
def level_order
queue = []
array = []
queue.push(@root)
while !(queue.empty?)
current_root = queue.first
queue.push(current_root.left) if !(current_root.left.nil?)
queue.push(current_root.right) if !(current_root.right.nil?)
if !(block_given?)
array << current_root.value
else
yield(current_root)
end
queue = queue[1..]
end
return array if !(block_given?)
end
def preorder(array = [], current_root=false, &block)
if current_root.nil? && block_given?
return
elsif current_root.nil? && !(block_given?)
return array
end
current_root = @root if current_root == false
if block_given?
yield(current_root)
else
array << current_root.value
end
preorder(array, current_root.left, &block)
preorder(array, current_root.right, &block)
end
def inorder(array = [], current_root=false, &block)
if current_root.nil? && block_given?
return
elsif current_root.nil? && !(block_given?)
return array
end
current_root = @root if current_root == false
inorder(array, current_root.left, &block)
if block_given?
yield(current_root)
else
array << current_root.value
end
inorder(array, current_root.right, &block)
end
def postorder(array = [], current_root=false, &block)
if current_root.nil? && block_given?
return
elsif current_root.nil? && !(block_given?)
return array
end
current_root = @root if current_root == false
postorder(array, current_root.left, &block)
postorder(array, current_root.right, &block)
if block_given?
yield(current_root)
else
array << current_root.value
end
end
def depth_(value, root = false, found = false, depth = 0)
root = @root if root == false
if !(root.nil?) && root.value == value
found = true
end
return puts "depth of #{value}: #{depth}" if root.nil? || found == true
if root.value < value
depth += 1
depth_(value, root.right, found, depth) if found == false
elsif root.value > value
depth += 1
depth_(value, root.left, found, depth) if found == false
end
end
def height(value, root = false, found = false, height_left = 0, height_right = 0)
root = @root if root == false
if !(root.nil?) && root.value == value
found = true
if root.left.nil? && root.right.nil?
return 0
end
if !(root.left.nil? && root.right.nil?)
current_root_left = root
until (current_root_left.left.nil?)
current_root_left = current_root_left.left
height_left += 1
end
current_root_right = root
until (current_root_right.right.nil?)
current_root_right = current_root_right.right
height_right += 1
height = true
end
end
if (height == true)
if height_left > height_right
height = height_left
elsif height_left < height_right
height = height_right
else
height = height_right
end
return height
end
end
if root.nil? || found == true && !(root.left.nil? && root.right.nil?)
return height
end
if root.value < value
height(value, root.right, found) if found == false
elsif root.value > value
height(value, root.left, found) if found == false
end
end
def balanced?(left_tree = false, right_tree = false, balanced = false, not_balanced = false)
left_tree = @root.left if left_tree == false
right_tree = @root.right if right_tree == false
left = height(left_tree.value)
right = height(right_tree.value)
if !(right.nil? && left.nil?)
if right > left && right <= left + 1 || left > right && left <= right + 1 || left == right
balanced = true
end
if !(right > left && right <= left + 1 || left > right && left <= right + 1 || left == right)
not_balanced = true
end
end
unless (right_tree.left.nil? && right_tree.right.nil?) && (left_tree.left.nil? && left_tree.right.nil?)
if not_balanced == false && balanced == true
puts 'balanced'
else
puts 'not balanced'
end
return
end
end
def rebalance(array = [], current_root = false)
if current_root.nil?
build_tree(array)
@array = array
return
end
current_root = @root if current_root == false
rebalance(array, current_root.left)
array << current_root.value
rebalance(array, current_root.right)
end
end
array = (Array.new(5) { rand(1..100) })
my_tree = Tree.new(array)
order_printing = lambda do
puts 'level order'
p my_tree.level_order
puts 'preorder'
p my_tree.preorder
puts 'postorder'
p my_tree.postorder
puts 'inorder'
p my_tree.inorder
end
print_balanced = lambda do
my_tree.pretty_print
my_tree.balanced?
end
my_tree.build_tree
print_balanced.call
order_printing.call
5.times { my_tree.insert(rand(100..150)) }
print_balanced.call
my_tree.rebalance
print_balanced.call
order_printing.call