Hello! Today we will learn about an important concept in graph theory called Minimum Spanning Tree (MST). A Minimum Spanning Tree is a tree that includes all connected vertices while minimizing the total weight of the edges. This concept is utilized in various fields such as network design, road connectivity, and clustering.
Problem Description
The problem is as follows:
Write a program to find the Minimum Spanning Tree of a given undirected weighted graph. The graph is given within the constraints
1 ≤ V ≤ 1000(number of vertices) and1 ≤ E ≤ 10000(number of edges), with all edge weights in the range1 ≤ w ≤ 1000.
Problem Solving Approach
There are several algorithms to find the Minimum Spanning Tree, but the two most commonly used are Prim’s Algorithm and Kruskal’s Algorithm. Here, we will describe how to solve the problem using Prim’s Algorithm.
Overview of Prim’s Algorithm
Prim’s Algorithm is an algorithm that proceeds by always selecting the edge with the minimum weight, following these steps:
- Select a starting vertex. (Any vertex can be chosen as the starting point.)
- Select the edge with the smallest weight among the edges connected to the currently selected vertex.
- Add the vertex connected by the selected edge to the tree.
- Repeat steps 2 and 3 until all vertices are included.
Time Complexity of Prim’s Algorithm
The time complexity of Prim’s Algorithm depends on the data structure used. Using a heap (priority queue) results in a complexity of O(E log V), while using an adjacency matrix results in a complexity of O(V2). Generally, using a heap is more efficient.
Implementation
Now let’s implement Prim’s Algorithm. Below is the Python code:
import heapq  # Importing to use heap
def prim(graph, start):
    mst = []  # List for the Minimum Spanning Tree
    visited = set()  # Set of visited vertices
    min_heap = [(0, start)]  # Initializing heap (weight, vertex)
    total_weight = 0  # Total weight
    while min_heap:
        weight, vertex = heapq.heappop(min_heap)  # Selecting edge with minimum weight
        if vertex not in visited:
            visited.add(vertex)  # Mark as visited
            total_weight += weight  # Summing weights
            mst.append((weight, vertex))
            for next_vertex, next_weight in graph[vertex].items():
                if next_vertex not in visited:
                    heapq.heappush(min_heap, (next_weight, next_vertex))  # Adding to heap
    return mst, total_weight  # Returning Minimum Spanning Tree and total weight
# Graph definition (adjacency list)
graph = {
    1: {2: 3, 3: 1},
    2: {1: 3, 3: 1, 4: 6},
    3: {1: 1, 2: 1, 4: 5},
    4: {2: 6, 3: 5}
}
mst, total_weight = prim(graph, 1)
print("Minimum Spanning Tree:", mst)
print("Total Weight:", total_weight)
    Code Explanation
The above code defines a function called prim. This function takes the graph and the starting vertex as arguments and returns the Minimum Spanning Tree along with its total weight.
- min_heap: Manages the currently selectable edges using a heap.
- visited: Tracks the selected vertices to prevent duplicates.
- After selecting the minimum weight edge from the heap, it adds the corresponding vertex to the tree and adds the connected vertices to the heap.
Running Test Cases
When you run the code above, you will get the following results:
Minimum Spanning Tree: [(0, 1), (1, 3), (1, 2)]
Total Weight: 2
    Here, the Minimum Spanning Tree includes vertices 1, 2, and 3, with a total weight of 2.
Complex Example
You can also find the Minimum Spanning Tree for more complex graphs using the same method. For example, consider a graph with multiple vertices and edges:
# Defining a complex graph
complex_graph = {
    'A': {'B': 4, 'H': 8},
    'B': {'A': 4, 'C': 8, 'H': 11},
    'C': {'B': 8, 'D': 7, 'F': 4, 'I': 2},
    'D': {'C': 7, 'E': 9, 'F': 14},
    'E': {'D': 9, 'F': 10},
    'F': {'C': 4, 'D': 14, 'E': 10, 'G': 2},
    'G': {'F': 2, 'H': 1, 'I': 6},
    'H': {'A': 8, 'B': 11, 'G': 1},
    'I': {'C': 2, 'G': 6}
}
mst_complex, total_weight_complex = prim(complex_graph, 'A')
print("Minimum Spanning Tree of the complex graph:", mst_complex)
print("Total Weight:", total_weight_complex)
    Result Interpretation
In this complex graph, multiple selections are needed to find the Minimum Spanning Tree. Prim’s Algorithm is effective in choosing optimal edges to construct the tree. The execution result displays the final Minimum Spanning Tree and its total weight.
Conclusion
The Minimum Spanning Tree plays an important role in network design and optimization problems. Through this tutorial, we have implemented Prim’s Algorithm in Python and applied it to real problems. Verifying the algorithm’s accuracy through various test cases is also an important process. I hope this helps you prepare for coding tests through problem-solving using algorithms!
Additional Learning Resources
Besides Prim’s Algorithm, it’s also beneficial to explore other methods like Kruskal’s Algorithm. Here are some resources to help understand the algorithms better: