Python Coding Test Course, Finding the Number of Connected Components

Hello, everyone! Today, I would like to discuss one of the frequently occurring problems in coding tests, which is “Counting Connected Components.” This problem involves determining how many connected components there are in a given graph, and it can be solved using graph traversal algorithms such as DFS (Depth-First Search) or BFS (Breadth-First Search). Through this lecture, we will learn how to understand the problem and find a solution.

Problem Description

The problem of counting connected components can be defined as follows:

Given a graph, count the number of connected components in this graph.

Input:
- The first line contains the number of vertices N (1 <= N <= 1000) and the number of edges M (0 <= M <= 10000).
- From the second line, M lines provide two integers representing the edge information. The two integers A and B indicate that vertex A is connected to vertex B.

Output:
- Output the number of connected components.

Problem Approach

To solve graph problems, it is important first to understand the structure of the graph. We can represent the graph based on the provided vertex and edge information by constructing an adjacency list. A connected component refers to a set of vertices that are connected to each other, and we use graph traversal algorithms to find them. As we visit all vertices in the graph, we can consider a new connected component discovered every time we encounter an unvisited vertex.

Implementation Steps

  1. Construct the graph in the form of an adjacency list from the input.
  2. Traverse the connected components using DFS or BFS while visiting all vertices.
  3. Increment the count of connected components every time an unvisited vertex is found during the traversal.

Code Implementation

Now, let's write the actual code based on the above algorithm. We will use Python to count the number of connected components using DFS.

def dfs(vertex, adjacency_list, visited):
    visited[vertex] = True # Mark the current vertex as visited
    for neighbor in adjacency_list[vertex]:
        if not visited[neighbor]:
            dfs(neighbor, adjacency_list, visited)

def count_connected_components(n, edges):
    adjacency_list = [[] for _ in range(n + 1)]
    for a, b in edges:
        adjacency_list[a].append(b)
        adjacency_list[b].append(a)

    visited = [False] * (n + 1)
    component_count = 0

    for vertex in range(1, n + 1):
        if not visited[vertex]:
            component_count += 1  # A new connected component is found
            dfs(vertex, adjacency_list, visited)

    return component_count

# Input
n, m = map(int, input().split())
edges = [tuple(map(int, input().split())) for _ in range(m)]

# Output the number of connected components
print(count_connected_components(n, edges))

Code Explanation

Looking at the above code, the following process is repeated.

  • First, we construct the graph in the form of an adjacency list. At this time, we create a list of size n + 1 to use indexes from 1 to n.
  • Using the given edge information, we form an undirected graph. For each vertex A, we add the connected vertex B and also add A to B to maintain bidirectionality.
  • We define a visited array for marking visited vertices. This stores whether each vertex has been visited.
  • We visit the vertices from 1 to n one by one, and when we find an unvisited vertex, we call DFS to explore all vertices connected to that vertex. During this process, we increment the count of connected components.

Complexity Analysis

The time complexity of this problem is O(N + M). Here, N represents the number of vertices, and M represents the number of edges. When performing DFS or BFS, each vertex and edge is visited once, which is the typical time complexity for graph traversal. The space complexity is also O(N + M), as we use an adjacency list representation of the graph and require additional arrays to check the visitation status.

Conclusion

We have discussed "Counting Connected Components" so far. This problem not only helps in understanding the basic concepts of graphs but can also be applied to various modified problems. I hope this lecture enhances your understanding of graphs. In the next lecture, we will tackle more interesting and useful algorithmic problems. Thank you.