In this article, we will explore one of the problems related to finding the ‘Lowest Common Ancestor (LCA)’ called ‘Finding the Lowest Common Ancestor 2’. This problem involves finding the lowest common ancestor of two nodes in a specific tree structure, particularly in a binary tree. The LCA problem is a fundamental problem associated with tree traversal and manipulation, frequently appearing in coding interviews and algorithm competitions.
Problem Description
Given a binary tree, we need to find the lowest common ancestor of two nodes u
and v
.
A binary tree can have a maximum of 2 children for each node, starting from the root node, and
each node has a unique value.
Input Format
- First line: Number of nodes
n
(1 ≤ n ≤ 100,000) - Second line: Tree structure (given parent nodes and child nodes)
- Third line: Two nodes
u
,v
Output Format
Output the lowest common ancestor of nodes u
and v
.
Example
Input: 7 1 2 1 3 2 4 2 5 3 6 3 7 4 5 Output: 2
Approach to the Problem
To solve this problem, we first need to construct the tree. The tree can be represented as an array or linked list, but
using a linked list is more efficient. Then, we will record the depth of each node through DFS (Depth First Search) and
store the parent nodes to easily find the LCA later.
Tree Construction
Based on the given edge information, we will establish the parent-child relationships for the nodes.
We will use Python’s dictionary to store the parent-child relationships.
from collections import defaultdict
def build_tree(edges):
tree = defaultdict(list)
for parent, child in edges:
tree[parent].append(child)
return tree
Storing Depth and Parent Node Information via DFS
def dfs(tree, node, parent, depth, depths, parents):
depths[node] = depth
parents[node] = parent
for child in tree[node]:
if child != parent:
dfs(tree, child, node, depth + 1, depths, parents)
Implementing the LCA Function
After aligning the positions of the two nodes based on depth, we proceed upward along their parents until
both nodes converge at the same node.
def lca(u, v, depths, parents):
# Aligning Depths
if depths[u] < depths[v]:
u, v = v, u
while depths[u] > depths[v]:
u = parents[u]
while u != v:
u = parents[u]
v = parents[v]
return u
Complete Code
def main():
n = int(input())
edges = [tuple(map(int, input().split())) for _ in range(n - 1)]
u, v = map(int, input().split())
tree = build_tree(edges)
depths = {}
parents = {}
# Collecting depth and parent information via DFS
dfs(tree, 1, -1, 0, depths, parents)
# Finding the lowest common ancestor
ancestor = lca(u, v, depths, parents)
print(ancestor)
if __name__ == "__main__":
main()
Time Complexity
The time complexity of the above algorithm is O(n). This is because each node is visited once during the tree construction and traversal.
Conclusion
The ‘Finding the Lowest Common Ancestor 2’ problem is helpful for understanding the basics of searching and manipulating binary trees,
and it is a practical algorithm. Through the process of solving this problem, one can gain a deep understanding of tree structures and the concept of DFS.
This algorithm can be extended to various other problems and is also very useful in the application of other data structures.