/
my_binary_tree.py
249 lines (184 loc) · 7.97 KB
/
my_binary_tree.py
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
# %%
class Node:
""" Node is user difine data structure. It is binary tree """
def __init__(self, data):
self.data = data
self.left = None
self.right = None
# %%
class BinaryTree:
""" user difine data structure BinaryTree """
def __init__(self, arr):
self.root = None # 今後の処理の基準にするために空のrootを作成する
for inserted_node_data in arr: #リストに格納されているノードの値を順次挿入する処理
print('....')
print('try inserting ', inserted_node_data)
self.insert(inserted_node_data)
def insert(self, data): # 挿入の処理(ルート生成 ⇒ 各ノードの枝を生成)・・・左枝がは現在ノードより小さい値が入る
if self.root == None: # ルートノードはtreeの解析の基準となる特別なノードなので.rootとして区別して生成するための場合分け
print('Root node is ....')
self.root = Node(data) # Node()インスタンスを生成して代入する
print(self.root.data)
else:
level = 1
flag = True
next_queue = [self.root] #初回のqueueを作成
while flag: #flagは要素が全てNoneのときFalseになる
temp_queue, next_queue = next_queue, []
level += 1
for node in temp_queue:
# 左分木
# 現在nodeのchiled nodeを次回操作のqueueに加えていく
if node.left is not None:
next_queue.append(node.left)
# child nodeにNoneが見つかったときには新たにdataを使ってnodeを生成する
else:
node.left = Node(data)
print('In level {}, {} is inseted'.format(level, data))
"""
(AA)
dataをnodeに追加したらこの回のinsertは終了する
ここはfor < while < insert methodの中なのでreturnを使って一発でメソッドを終了させる
"""
return
# 右分木
# 現在nodeのchiled nodeを次回操作のqueueに加えていく
if node.right is not None:
next_queue.append(node.right)
# child nodeにNoneが見つかったときには新たにdataを使ってnodeを生成する
else:
node.right = Node(data)
print('In level {}, {} is inseted'.format(level, data))
"""
(AA)参照
"""
return
flag = any(next_queue)
##########################
# Tree traversal
###############################
def preoder_traversal(self, node, res):
if node != None:
print('queue', node.data)
res.append(node.data)
# 前順序で左部分木
self.preoder_traversal(node.left, res)
# 前順序で右部分木
self.preoder_traversal(node.right, res)
return res
def inoder_traversal(self, node, res):
if node != None:
# 間順序で左部分木
self.inoder_traversal(node.left, res)
print('queue', node.data)
res.append(node.data)
# 間順序で右部分木
self.inoder_traversal(node.right, res)
return res
def postorder_traversal(self, node, res):
if node != None:
self.postorder_traversal(node.left, res)
self.postorder_traversal(node.right, res)
print('queue', node.data)
res.append(node.data)
return res
def level_order_traversal(self, queue, res= []):
if queue == [] :
# it's root
print('root', self.root.data)
res.append(self.root.data)
queue.append(self.root)
else:
# このlevelのノード は引数のqueueだからforで回す
temp_list, queue = queue, []
not_none_cnt = 0
for item in temp_list:
if item.left is not None:
res.append(item.left.data)
print('queue', item.left.data)
queue.append(item.left)
not_none_cnt += 1
if item.right is not None:
res.append(item.right.data)
print('queue', item.right.data)
queue.append(item.right)
not_none_cnt += 1
if not_none_cnt == 0:
return #最後にこの関数を呼び出したところに戻る
self.level_order_traversal(queue, res)
return res
# 二分木のメソッド
class BT_method(BinaryTree):
def __init__(self, arr):
super().__init__(arr)
def max_in_binary_tree(self, node, temp_max):
"""実装は親と子の関係で最大値を示す
traversしながらLIFOして、大きい値を残していっても同じ。
順探索でtraversしながら最大値を覚えておくのと同じこと"""
if node is not None:
temp_root_val = node.data
left_val = self.max_in_binary_tree(node.left, temp_max)
right_val = self.max_in_binary_tree(node.right, temp_max)
temp_max = max(temp_root_val, left_val, right_val, temp_max)
return temp_max
def find_val(self, node, val, flag=False):
if node != None:
if node.data == val:
return True
else:
flag_left = self.find_val(node.left, val) #再帰の結果をreturnで返しているので変数で受け取る
flag_right = self.find_val(node.right, val)
if flag_left or flag_right:
return True
return False
def size(self, node):
if node is None: #nodeがNoneのところで数え上げを終了
return 0 #0を返せば数え上げされない
else:
left_cnt = self.size(node.left)
right_cnt = self.size(node.right)
return 1 + left_cnt + right_cnt #自分(not None)が1と左右木中の数(仮想の探索を再帰関数で実現)
def hight(self, level=0):
flag = True
queue = [self.root]
while flag:
level += 1
temp_list, queue = queue, []
for node in temp_list:
if node.left is not None:
queue.append(node.left)
if node.right is not None:
queue.append(node.right)
flag = any(queue)
return level
def inoder_sort(self):
pass
def disorder_sort(self, parameter_list):
pass
ins = BinaryTree(range(1,16))
print('--------------------------')
print('start preoder traversal')
print(ins.preoder_traversal(ins.root, []))
print('--------------------------')
print('start inoder traversal')
print(ins.inoder_traversal(ins.root, []))
print('--------------------------')
print('start postoder traversal')
print(ins.postorder_traversal(ins.root, []))
print('--------------------------')
print('start level order traversal')
print(ins.level_order_traversal([]))
#
# print('=====================================')
ins2 = BT_method(range(1,16))
print('--------------------------')
print('find max')
print(ins2.max_in_binary_tree(ins2.root, 0))
print('--------------------------')
print('find value')
print('looking for 7', ins2.find_val(ins2.root, 7))
print('looking for 17', ins2.find_val(ins2.root, 17))
# search size
print('--------------------------')
print('detect node size')
print(ins2.size(ins2.root))