Skip to content

Commit ae40d42

Browse files
authored
Add files via upload
1 parent b8e1695 commit ae40d42

File tree

1 file changed

+141
-0
lines changed

1 file changed

+141
-0
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,141 @@
1+
# Lowest Common Ancestor in Binary Tree
2+
# Time Complexity : O(n) , Space Complexity : O(1)
3+
4+
"""
5+
Binary trees are trees where any node can have only 0, 1 or 2 children, which are sorted in any order.
6+
Lowest Common Ancestor (LCA) of two given nodes is the shared ancestor of n1 and n2 that is located farthest from the root
7+
or the last common node that occurs on the paths from root node to those nodes.
8+
9+
Here, we first check if both numbers exist in tree or not, with exists_in_tree() function by recursively traversing the trees.
10+
If yes then we find the LCA of both the elements with find_lca() function, which follows as :
11+
if LCA isnt found yet, but current node is one of the elements, then current node is the LCA
12+
else look for LCA in left and right subtree
13+
if both elements are found in different subtrees, make current node as LCA
14+
else if they are in same sub tree, return the LCA of that subtree
15+
16+
"""
17+
18+
19+
class Node :
20+
"""Class to define nodes of binary tree, with attributes data to store values, and left and right pointers"""
21+
def __init__(self, value) :
22+
self.data = value
23+
self.left = None
24+
self.right = None
25+
26+
def find_lca(node, n1, n2):
27+
"""Finds LCA of 2 numbers by recursively searching in left and right subtree
28+
input args : current node, the numbers whose lca is to be found, isFound list telling if the particular element is found
29+
returns : lca if found else None
30+
Time complexity : O(n), Space complexity : O(1)
31+
"""
32+
if node==None:
33+
return None
34+
35+
# if one element is found in current node, we return it
36+
# in normal case, both elements are found in different subtrees, called recursively, which is handeled later in this function
37+
# there may also be the case when one element is ancestor of another, which is handeled here, as this node is tha lca
38+
if node.data == n1:
39+
return node
40+
elif node.data == n2:
41+
return node
42+
43+
# Recursive calls
44+
lca_left = find_lca(node.left, n1, n2)
45+
lca_right = find_lca(node.right, n1, n2)
46+
47+
if lca_left and lca_right: # if both are not none, that is 1 element is found in each subtree
48+
return node # that means, current node is the lca
49+
else: # both numbers in same sub tree
50+
return lca_left if lca_left!=None else lca_right # return lca whichever is not none
51+
52+
53+
def exists_in_tree(node, n):
54+
"""This function finds if an element exists in a tree or not by recursively traversing the tree
55+
input : current node and the element to be found
56+
output : returns True if it exists, False if not
57+
Time complexity : O(n), Space complexity : O(1)
58+
"""
59+
if node == None:
60+
return False
61+
if node.data == n:
62+
return True
63+
64+
if exists_in_tree(node.left, n) or exists_in_tree(node.right, n):
65+
return True
66+
else:
67+
return False
68+
69+
70+
71+
def lowest_common_ancestor_binary_tree(root, n1, n2):
72+
"""This function checks if both elements are present in binary tree or not, and depending upon that returns proper value
73+
input : root node of the tree, numbers whose lca is to be found
74+
output : returns calculated LCA, if both exist, else None
75+
Both Time and Space complexity are : O(1)
76+
"""
77+
if exists_in_tree(root,n1) and exists_in_tree(root,n2): # if both the elements exist in the tree
78+
return find_lca(root, n1, n2) # then we calculate its LCA
79+
else :
80+
return None # otherwise we give answer None
81+
82+
83+
84+
def print_output(lca, n1, n2):
85+
"""Prints LCA of two numbers with a proper message if it exists"""
86+
if lca == None:
87+
print("\nElement does not exist in tree.\n")
88+
else:
89+
print(f"\nlca({n1}, {n2}) = {lca.data}\n")
90+
91+
92+
#============================= DRIVER CODE =============================
93+
if __name__ == "__main__":
94+
95+
root = Node(1)
96+
root.left = Node(2)
97+
root.right = Node(3)
98+
root.left.left = Node(4)
99+
root.left.right = Node(5)
100+
root.right.left = Node(6)
101+
root.right.right = Node(7)
102+
103+
"""
104+
1
105+
/ \
106+
2 3
107+
/ \ /\
108+
4 5 6 7
109+
"""
110+
111+
112+
lca = lowest_common_ancestor_binary_tree(root, 4, 5)
113+
print_output(lca, 4, 5)
114+
lca = lowest_common_ancestor_binary_tree(root, 2, 5)
115+
print_output(lca, 2, 5)
116+
lca = lowest_common_ancestor_binary_tree(root, 4, 2)
117+
print_output(lca, 4, 2)
118+
119+
120+
# OTHER EXAMPLES
121+
122+
"""lca = lowest_common_ancestor_binary_tree(root, 4, 7)
123+
print_output(lca, 4, 7)
124+
lca = lowest_common_ancestor_binary_tree(root, 2, 3)
125+
print_output(lca, 2, 3)
126+
lca = lowest_common_ancestor_binary_tree(root, 2, 1)
127+
print_output(lca, 2, 1)"""
128+
129+
"""lca = lowest_common_ancestor_binary_tree(root, 1, 8)
130+
print_output(lca, 1, 8)
131+
lca = lowest_common_ancestor_binary_tree(root, 1, 1)
132+
print_output(lca, 1, 1)
133+
lca = lowest_common_ancestor_binary_tree(root, 2, 6)
134+
print_output(lca, 2, 6)"""
135+
136+
"""lca = lowest_common_ancestor_binary_tree(root, 9, 8)
137+
print_output(lca, 9, 8)
138+
lca = lowest_common_ancestor_binary_tree(root, -1, 1)
139+
print_output(lca, -1, 1)
140+
lca = lowest_common_ancestor_binary_tree(root, -5, -6)
141+
print_output(lca, -5, -6)"""

0 commit comments

Comments
 (0)