684. Redundant Connection
Problem Description
In this problem, we're given a graph that was a tree initially, but an additional edge was added to it. An edge in this context is represented by a pair of nodes that it connects, given as [a, b]
. Since a proper tree has no cycles and is connected, the additional edge creates a single cycle in the graph. Our task is to identify and remove this redundant edge, restoring the graph to a tree. The graph is represented as an array edges
of n
edges, where each edge connects two different nodes from 1
to n
, and n
is the number of nodes in the tree. The goal is to find the redundant edge that can be removed to make the graph a tree again. If there are multiple redundant edges, we need to return the one that appears last in the edges
array. This problem checks the understanding of graph properties, specifically tree structures, and graph algorithms.
Intuition
The intuition behind this solution is based on the Union-Find algorithm, which is a classic data structure for keeping track of disjoint sets. The idea is to iterate through the list of edges and connect the nodes using the Union operation. We start by initializing a parent array p
, representing each node's parent in the disjoint set. Initially, each node is its own parent, indicating that they are each in separate sets.
As we process the edges, we use a Find operation, implemented as the find
function, to determine the set representative (parent) for each node. If the nodes connected by the current edge already have the same parent, it means that adding this edge would form a cycle, which is contrary to the tree definition. When we identify such an edge, we know it’s redundant and must be removed.
The clever part of the algorithm is the use of path compression in the Find operation. This optimization steps speeds up subsequent queries for the set representative by making each node along the path point to the ultimate parent directly.
Therefore, when we come across an edge connecting two nodes that are already in the same set (have the same parent), this edge is the redundant one, and we should return it as the answer. In the case that we process all edges without returning any, though highly unlikely for this problem's constraints, we return an empty list. However, since the problem states there's always one additional edge, we're guaranteed to find a redundant connection during our iteration through the edge list.
Learn more about Depth-First Search, Breadth-First Search, Union Find and Graph patterns.
Solution Approach
The solution approach involves the following steps, which are implemented with the Union-Find algorithm:
-
Initialization: We initialize an array
p
with1010
elements, which is assumed to be more than enough to cover the node labels up ton
. Here, each element is initialized to its index value, i.e.,p[i] = i
. This means each node starts as the parent of itself, representing a unique set for each node. -
Union-Find Algorithm:
-
The key operations in the Union-Find algorithm are
find
andunion
. -
Find Operation: This operation is implemented as a recursive function:
1def find(x): 2 if p[x] != x: 3 p[x] = find(p[x]) 4 return p[x]
The
find
function aims to locate the representative or the "parent" of the set that a node belongs to. If the node's parent is not itself, it proceeds to find the ultimate parent recursively and applies path compression by settingp[x]
to the ultimate parent. This optimizes the data structure so that subsequentfind
operations are faster. -
Union Operation: This operation is implicitly handled within the for loop:
1for a, b in edges: 2 if find(a) == find(b): 3 return [a, b] 4 p[find(a)] = find(b)
For each edge
(a, b)
inedges
, we first find the parent of botha
andb
. If they have the same parent, we found our redundant edge which is returned immediately because adding it to the tree would create a cycle. Otherwise, we perform the union by setting the parent of nodea
's set to be the parent of nodeb
's set, effectively merging the two sets.
-
-
Returning the Result:
-
The loop iterates over each edge only once and the redundant edge will be found during this iteration because we're guaranteed there is exactly one extra edge that creates a cycle.
-
The function
return [a, b]
statement is triggered when a redundant connection is found, which is the edge that causes the cycle. -
If the loop finishes without finding a redundant edge (which should not happen given the problem constraints), the function defaults to returning an empty list by
return []
.
-
In summary, the solution leverages the efficient Union-Find data structure with path compression to identify a cycle in the graph, which is the signature of the redundant connection in an otherwise acyclic tree structure.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Let's use a small example graph to illustrate the solution approach as described. Suppose our graph, which was originally a tree, has 4 nodes, and the edges
array is as follows: [[1, 2], [2, 3], [3, 1], [4, 3]]
. We can already see that there is a cycle formed by the edges [1, 2]
, [2, 3]
, and [3, 1]
. According to the problem statement, we need to find the redundant edge that appears last in the edges
list, which upon a preliminary look seems to be [3, 1]
.
-
Initialization:
We initialize our parent array
p
with enough space for our 4 nodes (in reality, our solution sets this up for more nodes, but for simplicity, here we'll limit it to the nodes we have). Thus,p = [0, 1, 2, 3, 4]
(we include0
just to match the node indices with their parent indices for convenience). -
Union-Find Algorithm:
We iterate through the
edges
array using thefind
andunion
operations as described:-
For the first edge
[1, 2]
, we find the parents of1
and2
, which are themselves, so we unite them by settingp[find(1)] = find(2)
. Nowp
becomes[0, 2, 2, 3, 4]
. -
For the second edge
[2, 3]
, we find the parents of2
and3
, which are2
and3
respectively. Since they are different, we unite them by settingp[find(2)] = find(3)
. Nowp
becomes[0, 3, 2, 3, 4]
and after path compression[0, 3, 3, 3, 4]
. -
For the third edge
[3, 1]
, we find the parents of3
and1
. Both are3
due to the path compression that has already occurred, indicating we've encountered a cycle. As this is the redundant edge we're looking for, we immediately return[3, 1]
. -
The fourth edge
[4, 3]
does not get processed as we've already found and returned the redundant edge in the step above.
-
-
Returning the Result:
-
We successfully find that
[3, 1]
is the redundant edge that creates a cycle in the graph. -
The function returns
[3, 1]
as the answer, which is the edge that must be removed to eliminate the cycle and restore the graph to a proper tree.
-
In this example, we can see that the Union-Find algorithm effectively identifies the edge causing a cycle due to the immediate parent check in the find
operation, assisted by the path compression that optimizes this check as we progress through the array of edges.
Solution Implementation
1from typing import List
2
3class Solution:
4 def findRedundantConnection(self, edges: List[List[int]]) -> List[int]:
5 # Function to find the set leader using path compression.
6 def find_set_leader(vertex):
7 # If the parent of the current vertex is not itself,
8 # recursively find the set leader and apply path compression.
9 if parent[vertex] != vertex:
10 parent[vertex] = find_set_leader(parent[vertex])
11 return parent[vertex]
12
13 # Initialize the parent array, with each vertex as its own set leader at the start.
14 parent = list(range(len(edges) + 1))
15
16 # Iterate through all the edges and apply union-find.
17 for a, b in edges:
18 # Find the set leaders for both vertices of the edge.
19 leader_a = find_set_leader(a)
20 leader_b = find_set_leader(b)
21
22 # If both vertices have the same set leader, a cycle is found, return the edge.
23 if leader_a == leader_b:
24 return [a, b]
25
26 # Union step: assign vertex b's leader to be the leader of vertex a's set.
27 parent[leader_a] = leader_b
28
29 # If no redundant connection found (no cycle), return an empty list.
30 return []
31
32# Example usage:
33# sol = Solution()
34# print(sol.findRedundantConnection([[1,2], [1,3], [2,3]])) # Output: [2, 3]
35
1class Solution {
2
3 // Array representing the parents of each node in the disjoint-set (union-find).
4 private int[] parent;
5
6 // Function to find the redundant connection, taking edges of a graph.
7 public int[] findRedundantConnection(int[][] edges) {
8
9 // Initialize the parent array for a maximum of 1010 nodes.
10 parent = new int[1010];
11 for (int i = 0; i < parent.length; i++) {
12 // Initially, each node is its own parent (self loop).
13 parent[i] = i;
14 }
15
16 // Iterate through all edges.
17 for (int[] edge : edges) {
18 int node1 = edge[0];
19 int node2 = edge[1];
20
21 // If both nodes have the same parent, the edge is redundant.
22 if (find(node1) == find(node2)) {
23 return edge;
24 }
25
26 // Union the sets of two nodes by setting the parent of one as the other.
27 parent[find(node1)] = find(node2);
28 }
29
30 // If no redundant connection is found, return null (should not happen according to the problem statement).
31 return null;
32 }
33
34 // Recursive function to find the parent of a given node.
35 private int find(int node) {
36
37 // Path compression: if the node is not its own parent, recursively find its parent and update the reference.
38 if (parent[node] != node) {
39 parent[node] = find(parent[node]);
40 }
41 // Return the parent of the node.
42 return parent[node];
43 }
44}
45
1#include <vector>
2using std::vector;
3
4class Solution {
5public:
6 // Parent array for Union-Find structure.
7 vector<int> parent;
8
9 // Function to find redundant connection in a graph represented as an edge list.
10 vector<int> findRedundantConnection(vector<vector<int>>& edges) {
11 // Initialize the parent array for Union-Find, with self-pointing indices.
12 parent.resize(1010);
13 for (int i = 0; i < parent.size(); ++i) parent[i] = i;
14
15 // Iterate over the edges.
16 for (auto& edge : edges) {
17 int nodeA = edge[0], nodeB = edge[1];
18 // Check if merging two sets creates a cycle (i.e., they have the same parent).
19 if (find(nodeA) == find(nodeB)) return edge; // If yes, then current edge is redundant.
20 // Union the two sets.
21 parent[find(nodeA)] = find(nodeB);
22 }
23 // If no redundant edge is found (which shouldn't happen), return an empty array.
24 return {};
25 }
26
27 // Helper function to find the root parent of a node, with path compression.
28 int find(int node) {
29 if (parent[node] != node) { // Check if node is not its own parent (not root node).
30 parent[node] = find(parent[node]); // Recursively find and set the root node.
31 }
32 return parent[node]; // Return the root node.
33 }
34};
35
1/**
2 * Function to find the redundant connection in a graph.
3 * Returns the edge that, when removed, would convert a graph into a tree (a graph with no cycles).
4 * The graph is represented as a list of edges.
5 *
6 * @param {number[][]} edges - A list of pairs representing the undirected edges of the graph.
7 * @return {number[]} - The redundant edge that forms a cycle in the graph.
8 */
9const findRedundantConnection = (edges: number[][]): number[] => {
10 // parent array to keep track of the root parent of each node in the graph.
11 const parent: number[] = Array.from({ length: 1010 }, (_, index) => index);
12
13 /**
14 * Helper function to find the root parent of a node.
15 * Implements path compression to optimize the union-find structure.
16 *
17 * @param {number} node - The node to find the root parent for.
18 * @return {number} - The root parent of the node.
19 */
20 function find(node: number): number {
21 if (parent[node] !== node) {
22 parent[node] = find(parent[node]);
23 }
24 return parent[node];
25 }
26
27 // Iterate through each edge to find the redundant connection.
28 for (let [nodeA, nodeB] of edges) {
29 // If the two nodes have the same root parent, a cycle is detected.
30 if (find(nodeA) === find(nodeB)) {
31 return [nodeA, nodeB];
32 }
33 // Union the sets.
34 parent[find(nodeA)] = find(nodeB);
35 }
36
37 // If no redundant connection is found, return an empty array.
38 return [];
39};
40
41export { findRedundantConnection }; // Optional: Use this if you want to import the function elsewhere.
42
Time and Space Complexity
The given code implements a Union-Find algorithm to detect a redundant connection in a graph. Let's analyze the time and space complexity:
Time Complexity
The time complexity of the Union-Find algorithm primarily depends on the two operations: find
and union
. Each edge triggers both operations once.
-
The
find
operation uses path compression, which gives us an amortized time complexity of almost constant time, "O(α(n))
", where α is the Inverse Ackermann function and is very slowly growing, therefore it's nearly constant for all practical purposes. -
The
union
takes "O(1)
" time since it's just assigning the root of one element to another.
Since the operations are performed for each edge and there are "E
" edges, the amortized time complexity of the algorithm is "O(Eα(n))
", which simplifies to "O(E)
" considering the very slow growth of α(n).
Space Complexity
For space complexity, we have:
- An array
p
, which stores the parents of each node in the graph. The size of the array is fixed at 1010, therefore, it requires "O(1)
" space since it's a constant space and doesn't depend on the size of the input.
Thus, the overall space complexity of the given code is "O(1)
".
Learn more about how to find time and space complexity quickly using problem constraints.
Given a sorted array of integers and an integer called target, find the element that
equals to the target and return its index. Select the correct code that fills the
___
in the given code snippet.
1def binary_search(arr, target):
2 left, right = 0, len(arr) - 1
3 while left ___ right:
4 mid = (left + right) // 2
5 if arr[mid] == target:
6 return mid
7 if arr[mid] < target:
8 ___ = mid + 1
9 else:
10 ___ = mid - 1
11 return -1
12
1public static int binarySearch(int[] arr, int target) {
2 int left = 0;
3 int right = arr.length - 1;
4
5 while (left ___ right) {
6 int mid = left + (right - left) / 2;
7 if (arr[mid] == target) return mid;
8 if (arr[mid] < target) {
9 ___ = mid + 1;
10 } else {
11 ___ = mid - 1;
12 }
13 }
14 return -1;
15}
16
1function binarySearch(arr, target) {
2 let left = 0;
3 let right = arr.length - 1;
4
5 while (left ___ right) {
6 let mid = left + Math.trunc((right - left) / 2);
7 if (arr[mid] == target) return mid;
8 if (arr[mid] < target) {
9 ___ = mid + 1;
10 } else {
11 ___ = mid - 1;
12 }
13 }
14 return -1;
15}
16
Recommended Readings
https algomonster s3 us east 2 amazonaws com cover_photos dfs svg Depth First Search Prereqs Recursion Review problems recursion_intro Trees problems tree_intro With a solid understanding of recursion under our belts we are now ready to tackle one of the most useful techniques in coding interviews Depth First Search DFS
https algomonster s3 us east 2 amazonaws com cover_photos bfs svg Breadth First Search on Trees Hopefully by this time you've drunk enough DFS Kool Aid to understand its immense power and seen enough visualization to create a call stack in your mind Now let me introduce the companion spell
Union Find Disjoint Set Union Data Structure Introduction Prerequisite Depth First Search Review problems dfs_intro Once we have a strong grasp of recursion and Depth First Search we can now introduce Disjoint Set Union DSU This data structure is motivated by the following problem Suppose we have sets of elements
Got a question? Ask the Monster Assistant anything you don't understand.
Still not clear? Submit the part you don't understand to our editors. Or join our Discord and ask the community.