Lesson 4
Discovering Connected Components in a Graph Using Depth-First Search
Introduction to the Lesson

Hello and welcome to the next lesson of our course, "Solving Algorithmic Problems with DFS." In previous lessons, we introduced Graphs and their representations, venturing into one of the most predominant procedures in Graph theory — Depth-First Search. Today, we'll learn how to use Depth-First Search to solve a common problem: determining if there is a cycle in the graph.

Understanding the Problem

Detecting cycles in a graph is a common problem that has applications in various domains. It is particularly useful in network routing, deadlock detection in operating systems, and in mathematical problems such as finding the presence of loops in a mathematical sequence.

In a graph, a cycle exists if we can start at a node, traverse along the edges, and return to the same node without retracing any edge. Our task is to construct a DFS function has_cycle(graph), which will check for a cycle in the given graph and return true if a cycle is found and false if the graph is acyclic.

Graphs can be represented in multiple ways, but for this lesson, we will use adjacency list representation for simplicity and efficiency.

An Efficient Solution using DFS

We traverse the graph starting from an initial node, and for every visited node, we check if it is being revisited during the DFS exploration. If it is, then a cycle has been detected. Hence, we return true. If no cycle is found after exploring all the nodes, we return false. This approach has a linear time complexity of O(V+E)O(V + E), where VV is the number of vertices (nodes) and EE is the number of edges in the graph.

Note: to avoid finding degenerate cycles (A -> B -> A), we will provide a parent vertex (the vertex we came from) to the dfs function on top of the current vertex we are at. This way, when reaching the next vertex from the current one, we first check if we're trying to reach the parent. If yes - we skip this edge, if not, and the vertex is already visited - we indeed found the cycle.

Building the Solution: DFS

The solution to this problem requires, first and foremost, implementing the DFS function. In this function, we mark the current node as visited and then check for each adjacent node. If the adjacent node is visited and it is not the parent of the current node, we find a cycle and return true. If the adjacent node is not visited, we recursively call the DFS function for that node.

Here is the modified dfs function for this task:

Python
1def dfs(vertex, visited, graph, parent): 2 visited.add(vertex) 3 for neighbor in graph[vertex]: 4 if neighbor not in visited: 5 if dfs(neighbor, visited, graph, vertex): 6 return True 7 elif neighbor != parent: 8 # The parent is already visited, but the parent -> vertex -> parent cycle is degenerate 9 return True 10 return False

The function adds the current vertex to visited nodes. It then explores neighbors of vertex. If the neighbor was not visited before, the function recursively visits the neighbor, specifying vertex as its parent. If the neighbor is already in the visited and it is not the parent of the current vertex, it means that a cycle has been found, and it returns True. If all the neighbor nodes are explored without finding a cycle, it returns False (indicating no cycle found from the vertex).

Building the Solution: Main Function

Then, we introduce a has_cycle function to wrap the dfs function call:

Python
1def has_cycle_connected(graph): 2 visited = set() 3 # Starting DFS from the first vertex in the graph 4 return dfs(next(iter(graph)), visited, graph, None)

Note that in case the graph is connected, it is enough to call the DFS function from any node just once. Do you have an idea how the above code should be changed to handle graphs consisting of more than one connected component?

Building the Solution: Complete Code Example

Here is the completed code with a simple test case:

Python
1def has_cycle_connected(graph): 2 visited = set() 3 # Starting DFS from the first vertex in the graph 4 return dfs(next(iter(graph)), visited, graph, None) 5 6def dfs(vertex, visited, stack, graph, parent): 7 visited.add(vertex) 8 9 for neighbor in graph[vertex]: 10 if neighbor not in visited: 11 if dfs(neighbor, visited, graph, vertex): 12 return True 13 elif neighbor != parent: 14 return True 15 16 return False 17 18 19graph = { 20 'A': ['B', 'C'], 21 'B': ['A', 'C'], 22 'C': ['A', 'B'], 23} 24print(has_cycle(graph)) 25# Output: True
Key Takeaways

In this lesson, you have learned how to use Depth-First Search to detect cycles in a graph. You now comprehend how to employ DFS to explore each path of a graph extensively, which is crucial for identifying cycles. Furthermore, you know how to leverage DFS to find optimized solutions for complex graph problems, achieving significantly lower time complexity than naive approaches. Ready to write some DFS? Let's go!

Enjoy this lesson? Now it's time to practice with Cosmo!
Practice is how you turn knowledge into actual skills.