CPSC 327 Data Structures and Algorithms Spring 2025

Programming Assignment 2 — Dijkstra's Algorithm
due Fri 3/14 in class


Graph ADT and AdjacencyListGraph

Dijkstra's algorithm is a graph algorithm, and is most efficient with an adjacency list implementation of the Graph ADT. You can either use your own implementation (if you implemented AdjacencyListGraph in programming assignment 1) or the provided implementation in graph.jar. To use the jar file, import it into your Eclipse project, right-click on it in the Package Explorer, and choose Build Path → Add to Build Path. To use your own implementation, copy it into your Eclipse project for this assignment.

Dijkstra's Algorithm

Implement the following version of Dijkstra's algorithm, a slight modification of the version discussed in class:

algorithm dijkstra(G,s):
  for all v in V do
    dist[v] ← ∞
    prev[v] = null

  dist[s] ← 0

  PQ ← makeQueue(V)

  while PQ is not empty do
    v ← PQ.removeMin()
    for each incident edge e=(v,u) 
      if dist[u] > dist[v]+w(v,u) then
        dist[u] = dist[v]+w(v,u)
        PQ.decreaseKey(u)
        prev[u] = e               // was prev[u] = v

The change is as indicated by the comment — instead of prev[u] storing the vertex immediately before u in the shortest path to u, it stores the edge coming into u in the shortest path to u.

Implement this algorithm in the init(s) and run(f) methods in Dijkstra.java, replacing the UnsupportedOperationException that is currently thrown. init should initialize the dist and prev labels and build the priority queue, while run should carry out the main while loop with one addition — run(f) only needs to find the shortest path to vertex f, so it should break out of the loop after vertex f is removed from the priority queue. (f == null means that the shortest paths to all vertices should be found so the loop runs until the priority queue is empty — does this case need special handling?)

Some notes:

Retrieving the Shortest Path

The remaining methods in Dijkstra.java (getPathVertices(f), getPathEdges(f), and getDist(f)) return information about the shortest path to vertex f:

The prev labels allow reconstruction of the shortest path to u by tracing backwards until the start is reached:

The start vertex can be identified because prev[s] == null. Storing edges instead of vertices makes getPathEdges easier to implement.

A few additional notes:

Testing

The main program Main.java provides a test case — it creates the graph shown, runs Dijkstra's algorithm to find the shortest paths from A to all other vertices, and prints out those paths and their lengths. Use it to test your implementation, and feel free to modify it however you want for further testing purposes.

The output of the provided main program should be:

A -> J: 10.0
  path vertices: A I J
  path edges: AI/9 IJ/1
A -> I: 9.0
  path vertices: A I
  path edges: AI/9
A -> H: 13.0
  path vertices: A D G H
  path edges: AD/4 DG/6 GH/3
A -> G: 10.0
  path vertices: A D G
  path edges: AD/4 DG/6
A -> F: 11.0
  path vertices: A B C F
  path edges: AB/6 BC/3 CF/2
A -> E: 7.0
  path vertices: A B E
  path edges: AB/6 BE/1
A -> D: 4.0
  path vertices: A D
  path edges: AD/4
A -> C: 9.0
  path vertices: A B C
  path edges: AB/6 BC/3
A -> B: 6.0
  path vertices: A B
  path edges: AB/6
A -> A: 0.0
  path vertices: A
  path edges:

About the Provided Code and Its Design

The Graph ADT only addresses the graph structure — it does not require elements like edge weights, nor does it prescribe how such additional information should be stored. In order to make the implementation of a graph algorithm like Dijkstra's algorithm as flexible as possible, it should similarly not make assumptions about how edge weights are stored. Instead, a "get weight" function that can be used when edge weights are desired is passed to the constructor — or, since Java doesn't truly support first-class functions, an object with a "get weight" function is passed to the constructor. (This is the same design principle behind the comparator object passed to PriorityQueue in order to allow different means of ordering the elements.)

"Object with a 'get weight' function" is implemented with the EdgeWeight interface in Dijkstra.java. The constructor's caller must then supply an instance of EdgeWeight. This can be done by defining a class that implements EdgeWeight and creating an instance of that class, but lambda expressions provide an alternative when the type is a functional interface (an interface containing a single subroutine). You can see the following in the provided Main:

  Dijkstra alg = new Dijkstra(g, e -> ((Integer) e.getObject()).doubleValue());

You don't need to know about lambda expressions in order to implement the methods in Dijkstra, but you can read more about them here if you are curious what the code above means.

Extra Credit

Address the following:

Handin

Hand in your code by copying your Java files into a folder called dijkstra within your handin directory (/classes/cs327/handin/username).

Hand in a hard copy of your writeup for the extra credit.