graph.util.MyDijkstraShortestPath.java Source code

Java tutorial

Introduction

Here is the source code for graph.util.MyDijkstraShortestPath.java

Source

package graph.util;

/*
* Copyright (c) 2003, the JUNG Project and the Regents of the University 
* of California
* All rights reserved.
*
* This software is open-source under the BSD license; see either
* "license.txt" or
* http://jung.sourceforge.net/license.txt for a description.
*/

import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;

import org.apache.commons.collections15.Transformer;

import edu.uci.ics.jung.algorithms.shortestpath.DijkstraDistance;
import edu.uci.ics.jung.algorithms.shortestpath.ShortestPath;
import edu.uci.ics.jung.graph.Graph;

/**
 * <p>Calculates distances and shortest paths using Dijkstra's   
 * single-source-shortest-path algorithm.  This is a lightweight
 * extension of <code>DijkstraDistance</code> that also stores
 * path information, so that the shortest paths can be reconstructed.</p>
 * 
 * <p> The elements in the maps returned by 
 * <code>getIncomingEdgeMap</code> are ordered (that is, returned 
 * by the iterator) by nondecreasing distance from <code>source</code>.</p>
 * 
 * @author Joshua O'Madadhain
 * @author Tom Nelson converted to jung2
 * @autor Lukas Sekerak edited for better performance
 * @see DijkstraDistance
 */
public class MyDijkstraShortestPath<V, E> extends DijkstraDistance<V, E> implements ShortestPath<V, E> {
    /**
     * <p>Creates an instance of <code>DijkstraShortestPath</code> for 
     * the specified graph and the specified method of extracting weights 
     * from edges, which caches results locally if and only if 
     * <code>cached</code> is <code>true</code>.
     * 
     * @param g     the graph on which distances will be calculated
     * @param nev   the class responsible for returning weights for edges
     * @param cached    specifies whether the results are to be cached
     */
    public MyDijkstraShortestPath(Graph<V, E> g, Transformer<E, ? extends Number> nev, boolean cached) {
        super(g, nev, cached);
    }

    /**
     * <p>Creates an instance of <code>DijkstraShortestPath</code> for 
     * the specified graph and the specified method of extracting weights 
     * from edges, which caches results locally.
     * 
     * @param g     the graph on which distances will be calculated
     * @param nev   the class responsible for returning weights for edges
     */
    public MyDijkstraShortestPath(Graph<V, E> g, Transformer<E, ? extends Number> nev) {
        super(g, nev);
    }

    /**
     * <p>Creates an instance of <code>DijkstraShortestPath</code> for 
     * the specified unweighted graph (that is, all weights 1) which
     * caches results locally.
     * 
     * @param g     the graph on which distances will be calculated
     */
    public MyDijkstraShortestPath(Graph<V, E> g) {
        super(g);
    }

    /**
     * <p>Creates an instance of <code>DijkstraShortestPath</code> for 
     * the specified unweighted graph (that is, all weights 1) which
     * caches results locally.
     * 
     * @param g     the graph on which distances will be calculated
     * @param cached    specifies whether the results are to be cached
     */
    public MyDijkstraShortestPath(Graph<V, E> g, boolean cached) {
        super(g, cached);
    }

    @Override
    protected SourceData getSourceData(V source) {
        SourceData sd = sourceMap.get(source);
        if (sd == null)
            sd = new SourcePathData(source);
        return sd;
    }

    /**
     * <p>Returns the last edge on a shortest path from <code>source</code>
     * to <code>target</code>, or null if <code>target</code> is not 
     * reachable from <code>source</code>.</p>
     * 
     * <p>If either vertex is not in the graph for which this instance
     * was created, throws <code>IllegalArgumentException</code>.</p>
     */
    public E getIncomingEdge(V source, V target) {
        if (!g.containsVertex(source))
            throw new IllegalArgumentException("Specified source vertex " + source + " is not part of graph " + g);

        if (!g.containsVertex(target))
            throw new IllegalArgumentException("Specified target vertex " + target + " is not part of graph " + g);

        Set<V> targets = new HashSet<V>();
        targets.add(target);
        singleSourceShortestPath(source, targets, g.getVertexCount());
        @SuppressWarnings("unchecked")
        Map<V, E> incomingEdgeMap = ((SourcePathData) sourceMap.get(source)).incomingEdges;
        E incomingEdge = incomingEdgeMap.get(target);

        if (!cached)
            reset(source);

        return incomingEdge;
    }

    /**
     * <p>Returns a <code>LinkedHashMap</code> which maps each vertex 
     * in the graph (including the <code>source</code> vertex) 
     * to the last edge on the shortest path from the 
     * <code>source</code> vertex.
     * The map's iterator will return the elements in order of 
     * increasing distance from <code>source</code>.</p>
     * 
     * @see DijkstraDistance#getDistanceMap(Object,int)
     * @see DijkstraDistance#getDistance(Object,Object)
     * @param source    the vertex from which distances are measured
     */
    public Map<V, E> getIncomingEdgeMap(V source) {
        return getIncomingEdgeMap(source, g.getVertexCount());
    }

    /**
     * Returns a <code>List</code> of the edges on the shortest path from 
     * <code>source</code> to <code>target</code>, in order of their
     * occurrence on this path.  
     * If either vertex is not in the graph for which this instance
     * was created, throws <code>IllegalArgumentException</code>.
     */
    public List<E> getPath(V source, V target) {
        if (!g.containsVertex(source))
            throw new IllegalArgumentException("Specified source vertex " + source + " is not part of graph " + g);

        if (!g.containsVertex(target))
            throw new IllegalArgumentException("Specified target vertex " + target + " is not part of graph " + g);

        LinkedList<E> path = new LinkedList<E>();

        // collect path data; must use internal method rather than
        // calling getIncomingEdge() because getIncomingEdge() may
        // wipe out results if results are not cached
        Set<V> targets = new HashSet<V>();
        targets.add(target);
        singleSourceShortestPath(source, targets, g.getVertexCount());
        @SuppressWarnings("unchecked")
        Map<V, E> incomingEdges = ((SourcePathData) sourceMap.get(source)).incomingEdges;

        if (incomingEdges.isEmpty() || incomingEdges.get(target) == null)
            return path;
        V current = target;
        while (!current.equals(source)) {
            E incoming = incomingEdges.get(current);
            path.addFirst(incoming);
            current = ((Graph<V, E>) g).getOpposite(current, incoming);
        }
        return path;
    }

    /**
     * <p>Returns a <code>LinkedHashMap</code> which maps each of the closest 
     * <code>numDist</code> vertices to the <code>source</code> vertex 
     * in the graph (including the <code>source</code> vertex) 
     * to the incoming edge along the path from that vertex.  Throws 
     * an <code>IllegalArgumentException</code> if <code>source</code>
     * is not in this instance's graph, or if <code>numDests</code> is 
     * either less than 1 or greater than the number of vertices in the 
     * graph.
     * 
     * @see #getIncomingEdgeMap(Object)
     * @see #getPath(Object,Object)
     * @param source    the vertex from which distances are measured
     * @param numDests  the number of vertics for which to measure distances
     */
    public LinkedHashMap<V, E> getIncomingEdgeMap(V source, int numDests) {
        if (g.getVertices().contains(source) == false)
            throw new IllegalArgumentException("Specified source vertex " + source + " is not part of graph " + g);

        if (numDests < 1 || numDests > g.getVertexCount())
            throw new IllegalArgumentException("numDests must be >= 1 " + "and <= g.numVertices()");

        singleSourceShortestPath(source, null, numDests);

        @SuppressWarnings("unchecked")
        LinkedHashMap<V, E> incomingEdgeMap = ((SourcePathData) sourceMap.get(source)).incomingEdges;

        if (!cached)
            reset(source);

        return incomingEdgeMap;
    }

    /**
     * For a given source vertex, holds the estimated and final distances, 
     * tentative and final assignments of incoming edges on the shortest path from
     * the source vertex, and a priority queue (ordered by estimaed distance)
     * of the vertices for which distances are unknown.
     * 
     * @author Joshua O'Madadhain
     */
    protected class SourcePathData extends SourceData {
        protected Map<V, E> tentativeIncomingEdges;
        protected LinkedHashMap<V, E> incomingEdges;

        protected SourcePathData(V source) {
            super(source);
            incomingEdges = new LinkedHashMap<V, E>();
            tentativeIncomingEdges = new HashMap<V, E>();
        }

        @Override
        public void update(V dest, E tentative_edge, double new_dist) {
            super.update(dest, tentative_edge, new_dist);
            tentativeIncomingEdges.put(dest, tentative_edge);
        }

        @Override
        public Map.Entry<V, Number> getNextVertex() {
            Map.Entry<V, Number> p = super.getNextVertex();
            V v = p.getKey();
            E incoming = tentativeIncomingEdges.remove(v);
            incomingEdges.put(v, incoming);
            return p;
        }

        @Override
        public void restoreVertex(V v, double dist) {
            super.restoreVertex(v, dist);
            E incoming = incomingEdges.get(v);
            tentativeIncomingEdges.put(v, incoming);
        }

        @Override
        public void createRecord(V w, E e, double new_dist) {
            super.createRecord(w, e, new_dist);
            tentativeIncomingEdges.put(w, e);
        }

    }

    /**
     * Do zoznamu sa pridaju vsetke vrcholy ktore su na ceste
     * medzi source a target, okrem source vrchola.  
     *  
    * @author Joshua O'Madadhain
    * @author Tom Nelson converted to jung2
    * @autor Lukas Sekerak edited
    * @see DijkstraDistance
    */
    public void getPathVertices(V source, V target, Set<V> verticesInPath) {
        Set<V> targets = new HashSet<V>();
        targets.add(target);
        singleSourceShortestPath(source, targets, g.getVertexCount());
        @SuppressWarnings("unchecked")
        Map<V, E> incomingEdges = ((SourcePathData) sourceMap.get(source)).incomingEdges;

        if (incomingEdges.isEmpty() || incomingEdges.get(target) == null)
            return;
        V current = target;
        while (!current.equals(source)) {
            E incoming = incomingEdges.get(current);
            verticesInPath.add(current);
            current = ((Graph<V, E>) g).getOpposite(current, incoming);
        }
    }
}