Skip to content

Commit f680806

Browse files
add type hints for avl_tree (#4214)
Co-authored-by: LiHao <[email protected]>
1 parent 2a6e4bb commit f680806

File tree

1 file changed

+87
-68
lines changed

1 file changed

+87
-68
lines changed

Diff for: data_structures/binary_tree/avl_tree.py

+87-68
Original file line numberDiff line numberDiff line change
@@ -8,84 +8,85 @@
88

99
import math
1010
import random
11+
from typing import Any, List, Optional
1112

1213

1314
class my_queue:
14-
def __init__(self):
15-
self.data = []
16-
self.head = 0
17-
self.tail = 0
15+
def __init__(self) -> None:
16+
self.data: List[Any] = []
17+
self.head: int = 0
18+
self.tail: int = 0
1819

19-
def is_empty(self):
20+
def is_empty(self) -> bool:
2021
return self.head == self.tail
2122

22-
def push(self, data):
23+
def push(self, data: Any) -> None:
2324
self.data.append(data)
2425
self.tail = self.tail + 1
2526

26-
def pop(self):
27+
def pop(self) -> Any:
2728
ret = self.data[self.head]
2829
self.head = self.head + 1
2930
return ret
3031

31-
def count(self):
32+
def count(self) -> int:
3233
return self.tail - self.head
3334

34-
def print(self):
35+
def print(self) -> None:
3536
print(self.data)
3637
print("**************")
3738
print(self.data[self.head : self.tail])
3839

3940

4041
class my_node:
41-
def __init__(self, data):
42+
def __init__(self, data: Any) -> None:
4243
self.data = data
43-
self.left = None
44-
self.right = None
45-
self.height = 1
44+
self.left: Optional[my_node] = None
45+
self.right: Optional[my_node] = None
46+
self.height: int = 1
4647

47-
def get_data(self):
48+
def get_data(self) -> Any:
4849
return self.data
4950

50-
def get_left(self):
51+
def get_left(self) -> Optional["my_node"]:
5152
return self.left
5253

53-
def get_right(self):
54+
def get_right(self) -> Optional["my_node"]:
5455
return self.right
5556

56-
def get_height(self):
57+
def get_height(self) -> int:
5758
return self.height
5859

59-
def set_data(self, data):
60+
def set_data(self, data: Any) -> None:
6061
self.data = data
6162
return
6263

63-
def set_left(self, node):
64+
def set_left(self, node: Optional["my_node"]) -> None:
6465
self.left = node
6566
return
6667

67-
def set_right(self, node):
68+
def set_right(self, node: Optional["my_node"]) -> None:
6869
self.right = node
6970
return
7071

71-
def set_height(self, height):
72+
def set_height(self, height: int) -> None:
7273
self.height = height
7374
return
7475

7576

76-
def get_height(node):
77+
def get_height(node: Optional["my_node"]) -> int:
7778
if node is None:
7879
return 0
7980
return node.get_height()
8081

8182

82-
def my_max(a, b):
83+
def my_max(a: int, b: int) -> int:
8384
if a > b:
8485
return a
8586
return b
8687

8788

88-
def right_rotation(node):
89+
def right_rotation(node: my_node) -> my_node:
8990
r"""
9091
A B
9192
/ \ / \
@@ -98,6 +99,7 @@ def right_rotation(node):
9899
"""
99100
print("left rotation node:", node.get_data())
100101
ret = node.get_left()
102+
assert ret is not None
101103
node.set_left(ret.get_right())
102104
ret.set_right(node)
103105
h1 = my_max(get_height(node.get_right()), get_height(node.get_left())) + 1
@@ -107,12 +109,13 @@ def right_rotation(node):
107109
return ret
108110

109111

110-
def left_rotation(node):
112+
def left_rotation(node: my_node) -> my_node:
111113
"""
112114
a mirror symmetry rotation of the left_rotation
113115
"""
114116
print("right rotation node:", node.get_data())
115117
ret = node.get_right()
118+
assert ret is not None
116119
node.set_right(ret.get_left())
117120
ret.set_left(node)
118121
h1 = my_max(get_height(node.get_right()), get_height(node.get_left())) + 1
@@ -122,7 +125,7 @@ def left_rotation(node):
122125
return ret
123126

124127

125-
def lr_rotation(node):
128+
def lr_rotation(node: my_node) -> my_node:
126129
r"""
127130
A A Br
128131
/ \ / \ / \
@@ -133,33 +136,41 @@ def lr_rotation(node):
133136
UB Bl
134137
RR = right_rotation LR = left_rotation
135138
"""
136-
node.set_left(left_rotation(node.get_left()))
139+
left_child = node.get_left()
140+
assert left_child is not None
141+
node.set_left(left_rotation(left_child))
137142
return right_rotation(node)
138143

139144

140-
def rl_rotation(node):
141-
node.set_right(right_rotation(node.get_right()))
145+
def rl_rotation(node: my_node) -> my_node:
146+
right_child = node.get_right()
147+
assert right_child is not None
148+
node.set_right(right_rotation(right_child))
142149
return left_rotation(node)
143150

144151

145-
def insert_node(node, data):
152+
def insert_node(node: Optional["my_node"], data: Any) -> Optional["my_node"]:
146153
if node is None:
147154
return my_node(data)
148155
if data < node.get_data():
149156
node.set_left(insert_node(node.get_left(), data))
150157
if (
151158
get_height(node.get_left()) - get_height(node.get_right()) == 2
152159
): # an unbalance detected
160+
left_child = node.get_left()
161+
assert left_child is not None
153162
if (
154-
data < node.get_left().get_data()
163+
data < left_child.get_data()
155164
): # new node is the left child of the left child
156165
node = right_rotation(node)
157166
else:
158167
node = lr_rotation(node)
159168
else:
160169
node.set_right(insert_node(node.get_right(), data))
161170
if get_height(node.get_right()) - get_height(node.get_left()) == 2:
162-
if data < node.get_right().get_data():
171+
right_child = node.get_right()
172+
assert right_child is not None
173+
if data < right_child.get_data():
163174
node = rl_rotation(node)
164175
else:
165176
node = left_rotation(node)
@@ -168,52 +179,59 @@ def insert_node(node, data):
168179
return node
169180

170181

171-
def get_rightMost(root):
172-
while root.get_right() is not None:
173-
root = root.get_right()
182+
def get_rightMost(root: my_node) -> Any:
183+
while True:
184+
right_child = root.get_right()
185+
if right_child is None:
186+
break
187+
root = right_child
174188
return root.get_data()
175189

176190

177-
def get_leftMost(root):
178-
while root.get_left() is not None:
179-
root = root.get_left()
191+
def get_leftMost(root: my_node) -> Any:
192+
while True:
193+
left_child = root.get_left()
194+
if left_child is None:
195+
break
196+
root = left_child
180197
return root.get_data()
181198

182199

183-
def del_node(root, data):
200+
def del_node(root: my_node, data: Any) -> Optional["my_node"]:
201+
left_child = root.get_left()
202+
right_child = root.get_right()
184203
if root.get_data() == data:
185-
if root.get_left() is not None and root.get_right() is not None:
186-
temp_data = get_leftMost(root.get_right())
204+
if left_child is not None and right_child is not None:
205+
temp_data = get_leftMost(right_child)
187206
root.set_data(temp_data)
188-
root.set_right(del_node(root.get_right(), temp_data))
189-
elif root.get_left() is not None:
190-
root = root.get_left()
207+
root.set_right(del_node(right_child, temp_data))
208+
elif left_child is not None:
209+
root = left_child
210+
elif right_child is not None:
211+
root = right_child
191212
else:
192-
root = root.get_right()
213+
return None
193214
elif root.get_data() > data:
194-
if root.get_left() is None:
215+
if left_child is None:
195216
print("No such data")
196217
return root
197218
else:
198-
root.set_left(del_node(root.get_left(), data))
199-
elif root.get_data() < data:
200-
if root.get_right() is None:
219+
root.set_left(del_node(left_child, data))
220+
else: # root.get_data() < data
221+
if right_child is None:
201222
return root
202223
else:
203-
root.set_right(del_node(root.get_right(), data))
204-
if root is None:
205-
return root
206-
if get_height(root.get_right()) - get_height(root.get_left()) == 2:
207-
if get_height(root.get_right().get_right()) > get_height(
208-
root.get_right().get_left()
209-
):
224+
root.set_right(del_node(right_child, data))
225+
226+
if get_height(right_child) - get_height(left_child) == 2:
227+
assert right_child is not None
228+
if get_height(right_child.get_right()) > get_height(right_child.get_left()):
210229
root = left_rotation(root)
211230
else:
212231
root = rl_rotation(root)
213-
elif get_height(root.get_right()) - get_height(root.get_left()) == -2:
214-
if get_height(root.get_left().get_left()) > get_height(
215-
root.get_left().get_right()
216-
):
232+
elif get_height(right_child) - get_height(left_child) == -2:
233+
assert left_child is not None
234+
if get_height(left_child.get_left()) > get_height(left_child.get_right()):
217235
root = right_rotation(root)
218236
else:
219237
root = lr_rotation(root)
@@ -256,25 +274,26 @@ class AVLtree:
256274
*************************************
257275
"""
258276

259-
def __init__(self):
260-
self.root = None
277+
def __init__(self) -> None:
278+
self.root: Optional[my_node] = None
261279

262-
def get_height(self):
263-
# print("yyy")
280+
def get_height(self) -> int:
264281
return get_height(self.root)
265282

266-
def insert(self, data):
283+
def insert(self, data: Any) -> None:
267284
print("insert:" + str(data))
268285
self.root = insert_node(self.root, data)
269286

270-
def del_node(self, data):
287+
def del_node(self, data: Any) -> None:
271288
print("delete:" + str(data))
272289
if self.root is None:
273290
print("Tree is empty!")
274291
return
275292
self.root = del_node(self.root, data)
276293

277-
def __str__(self): # a level traversale, gives a more intuitive look on the tree
294+
def __str__(
295+
self,
296+
) -> str: # a level traversale, gives a more intuitive look on the tree
278297
output = ""
279298
q = my_queue()
280299
q.push(self.root)
@@ -308,7 +327,7 @@ def __str__(self): # a level traversale, gives a more intuitive look on the tre
308327
return output
309328

310329

311-
def _test():
330+
def _test() -> None:
312331
import doctest
313332

314333
doctest.testmod()

0 commit comments

Comments
 (0)