Click here to Skip to main content
6,595,854 members and growing! (18,530 online)
Email Password   helpLost your password?
General Programming » Algorithms & Recipes » Path Finding     Intermediate License: The Code Project Open License (CPOL)

Dijkstra:Shortest Route Calculation - Object Oriented

By Michael Demeersseman

Calculation of the shortest path between x points
C#, Windows, .NET (.NET 3.5), Dev
Posted:3 Jan 2008
Views:38,423
Bookmarked:42 times
Announcements
Loading...
 
Search    
Advanced Search
Add to IE Search
printPrint   add Share
      Discuss Discuss   Broken Article?Report  
13 votes for this article.
Popularity: 4.15 Rating: 3.73 out of 5
1 vote, 7.7%
1
2 votes, 15.4%
2

3
5 votes, 38.5%
4
5 votes, 38.5%
5
All points are locations. The connections between the points have a specific weight. Not all connections are bidirectional (a dot marks a start travel point). When Calculate is pressed, all routes from the selected location are calculated. When a route is selected in the listbox, the shortest route is visually shown by coloring the start dots red.
In this example, the shortest route from 0 to 4 is going through location 2, 1 and then 4.

Introduction

Dijkstra was a Dutch computer scientist who invented a fast and simple way to calculate the shortest path between two points. Many examples I have found on the Internet implement that algorithm but none of them have done it in an Object Oriented way. So I thought of making my own.

Background

Information about Dijkstra can be found here.

Using the Code

The code contains two Project classes:

  1. GUI: Shows the information visually
    • To add locations, click on the 'Add Location' button and then click on the map where you want to add locations.
    • To add routes, click on the 'Add Location' button to deactivate the add location, then click on a start location, then click on a end location. The weight of the route can be configured on top.
  2. RouteEngine: Calculates the route

I will only go into details about the RouteEngine. How the UI is handled is not so important for this article but if you need information about it, you can always ask.

Project RouteEngine

  1. Connection: This class holds the information about the connection between two dots. This is a one directional connection from A (the startpoint is visually shown with a dot) to B with a specific weight attached.
  2. Location: Just a location (for example 1).
  3. RouteEngine: This class will calculate all routes from one given startPoint.
  4. Route: This class holds the information about a route between two points (generated with the RouteEngine class).

Location

The most simple class. It only holds a name to display.

Connection

This class contains two Location objects and a weight.

public Connection(Location a, Location b, int weight)
{
    this._a = a;
    this._b = b;
    this._weight = weight;
}

Route

This class contains a route. It has only a list of connections and the total weight. This class is generated by the route engine.

Route Engine

This is the class that drives the component. The algorithm is as follows:

  1. Set the startPosition as active
  2. Set the total weight to all routes to infinite
  3. Iterate through all connections of the active position and store their weight if their weight is smaller than their current weight
  4. Set the active position as used
  5. Set the nearest point (on whatever location) that isn't used as active
  6. Repeat 3, 4, 5 until all positions are used

The following method will perform all these steps (and some extra checking and thinking). The Dictionary returned is a list of destination locations and the corresponding route to each destination location.

/// 
/// Calculates the shortest route to all the other locations
/// 
/// List of all locations and their shortest route
public Dictionary CalculateMinCost(Location _startLocation)
{
    //Initialise a new empty route list
    Dictionary _shortestPaths = new Dictionary();
    //Initialise a new empty handled locations list
    List _handledLocations = new List();

    //Initialise the new routes. the constructor will set the route weight to in.max
    foreach (Location location in _locations)
    {
        _shortestPaths.Add(location, new Route(location.Identifier));
    }

    //The startPosition has a weight 0.
    _shortestPaths[_startLocation].Cost = 0;

    //If all locations are handled, stop the engine and return the result
    while (_handledLocations.Count != _locations.Count)
    {
        //Order the locations
        List _shortestLocations = (List < Location > )(from s in _shortestPaths
                                orderby s.Value.Cost
                                select s.Key).ToList();

        Location _locationToProcess = null;

        //Search for the nearest location that isn't handled
        foreach (Location _location in _shortestLocations)
        {
            if (!_handledLocations.Contains(_location))
            {
                //If the cost equals int.max, there are no more possible connections 
                //to the remaining locations
                if (_shortestPaths[_location].Cost == int.MaxValue)
                    return _shortestPaths;
                _locationToProcess = _location;
                break;
            }
        }

        //Select all connections where the startposition is the location to Process
        var _selectedConnections = from c in _connections
                                   where c.A == _locationToProcess
                                   select c;

        //Iterate through all connections and search for a connection which is shorter
        foreach (Connection conn in _selectedConnections)
        {
            if (_shortestPaths[conn.B].Cost > conn.Weight + _shortestPaths[conn.A].Cost)
            {
                _shortestPaths[conn.B].Connections = 
                        _shortestPaths[conn.A].Connections.ToList();
                _shortestPaths[conn.B].Connections.Add(conn);
                _shortestPaths[conn.B].Cost = conn.Weight + _shortestPaths[conn.A].Cost;
            }
        }
        //Add the location to the list of processed locations
        _handledLocations.Add(_locationToProcess);
    }

    return _shortestPaths;
}

History

  • 24 December, 2007: First release

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)

About the Author

Michael Demeersseman


Member

Location: Belgium Belgium

Other popular Algorithms & Recipes articles:

Article Top
You must Sign In to use this message board.
FAQ FAQ 
 
Noise Tolerance  Layout  Per page   
 Msgs 1 to 9 of 9 (Total in Forum: 9) (Refresh)FirstPrevNext
QuestionInstallation on Windows Vista Pinmemberssnc9:38 17 Mar '09  
QuestionFile 'usrLocation.cs' not found. PinmemberMember 330569820:52 22 Apr '08  
AnswerRe: File 'usrLocation.cs' not found. PinmemberMichael Demeersseman21:58 23 May '08  
GeneralBug! PinmemberManjit Dosanjh0:46 10 Jan '08  
GeneralRe: Bug! PinmemberMichael Demeersseman10:24 10 Jan '08  
GeneralRe: Bug! PinmemberManjit Dosanjh1:03 11 Jan '08  
GeneralRe: Bug! PinmemberMichael Demeersseman21:58 23 May '08  
QuestionWhat version? PinmemberManjit Dosanjh12:40 9 Jan '08  
GeneralRe: What version? PinmemberManjit Dosanjh0:43 10 Jan '08  

General General    News News    Question Question    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

PermaLink | Privacy | Terms of Use
Last Updated: 3 Jan 2008
Editor: Sean Ewington
Copyright 2008 by Michael Demeersseman
Everything else Copyright © CodeProject, 1999-2009
Web18 | Advertise on the Code Project