You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

62 lines
2.0 KiB
Java

package com.jwetherell.algorithms.graph;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.PriorityQueue;
import java.util.Queue;
import java.util.Set;
import com.jwetherell.algorithms.data_structures.Graph;
/**
* Prim's minimum spanning tree. Only works on undirected graphs. It finds a
* subset of the edges that forms a tree that includes every vertex, where the
* total weight of all the edges in the tree is minimized.
*
* https://en.wikipedia.org/wiki/Prim%27s_algorithm
*
* @author Justin Wetherell <phishman3579@gmail.com>
*/
public class Prim {
private Prim() { }
public static Graph.CostPathPair<Integer> getMinimumSpanningTree(Graph<Integer> graph, Graph.Vertex<Integer> start) {
if (graph == null)
throw (new NullPointerException("Graph must be non-NULL."));
// Prim's algorithm only works on undirected graphs
if (graph.getType() == Graph.TYPE.DIRECTED)
throw (new IllegalArgumentException("Undirected graphs only."));
int cost = 0;
final Set<Graph.Vertex<Integer>> unvisited = new HashSet<Graph.Vertex<Integer>>();
unvisited.addAll(graph.getVertices());
unvisited.remove(start); // O(1)
final List<Graph.Edge<Integer>> path = new ArrayList<Graph.Edge<Integer>>();
final Queue<Graph.Edge<Integer>> edgesAvailable = new PriorityQueue<Graph.Edge<Integer>>();
Graph.Vertex<Integer> vertex = start;
while (!unvisited.isEmpty()) {
// Add all edges to unvisited vertices
for (Graph.Edge<Integer> e : vertex.getEdges()) {
if (unvisited.contains(e.getToVertex()))
edgesAvailable.add(e);
}
// Remove the lowest cost edge
final Graph.Edge<Integer> e = edgesAvailable.remove();
cost += e.getCost();
path.add(e); // O(1)
vertex = e.getToVertex();
unvisited.remove(vertex); // O(1)
}
return (new Graph.CostPathPair<Integer>(cost, path));
}
}