Javascript/Algorithms

Dijkstra's Algorithm

laughcryrepeat 2021. 8. 15. 00:24

one of the most famous and widely used algorithms

the shortest path between the  vertices.

 

Edsger Dijkstra, a Dutch programmer

 

Useful for

 

  • GPS - finding fastest route
  • Network Routing - finds open shortest path for data
  • Biology - used to model the spread of viruses among humans
  • Airline tickets - finding cheapest route to your destination
  • Many other uses!

 

thr first thing is we need to do is adding a weight (associated value) to the graph.

class WeightedGraph {
  constructor() {
    this.adjacencyList = {};
  }
  addVertex(vertex) {
    if(!this.adjacencyList[vertex]) this.adjacecyList[vertex] = [];
  }
  addEdge(vertex1, vertex2, weight) {
    this.adjacencyList[vertex1].push({node:vertex2, weight});
    this.adjacencyList[vertex2].push({node:vertex1, weight});
  }
}

 

Find the shortest path from A to E

 

 

Dijkstra's algorithm works not only to give the shortest path between two nodes.

at the end, the way of implementing it, we'll have a data structure that gives the shortest path from A to all the nodes.

 

 

A priority queue how that played a role in the algorithm

to find the next smallest one to visit.

 

everytime you add something to it, you give it a priority.

and get the smallest out first.

class PriorityQueue {
  constructor() {
    this.values = [];
  }
  enqueue(val, priority) {
    this.values.push({val, priority});
    this.sort();
  }
  dequeue() {
    return this.values.shift();
  }
  sort() {
    this.values.sort((a,b) => a.priority - b.priority);
  }
}

 

Dijkstra's Pseudocode

 

  • This function should accept a starting and ending vertex
  • Create an object (we'll call it distances) and set each key to be every vertex in the adjacency list with a value of infinity, except for the starting vertex which should have a value of 0.
  • After setting a value in the distances object, add each vertex with a priority of Infinity to the priority queue, except the starting vertex, which should have a priority of 0 because that's where we begin.
  • Create another object called previous and set each key to be every vertex in the adjacency list with a value of null
  • Start looping as long as there is anything in the priority queue
    • dequeue a vertex from the priority queue
    • If that vertex is the same as the ending vertex - we are done!
    • Otherwise loop through each value in the adjacency list at that vertex
      • Calculate the distance to that vertex from the starting vertex
      • if the distance is less than what is currently stored in our distances object
        • update the distances object with new lower distance
        • update the previous object to contain that vertex
        • enqueue the vertex with the total distance from the start node

 

class WeightedGraph {
  constructor() {
    this.adjacencyList = {};
  }
  addVertex(vertex) {
    if(!this.adjacencyList[vertex]) this.adjacecyList[vertex] = [];
  }
  addEdge(vertex1, vertex2, weight) {
    this.adjacencyList[vertex1].push({node:vertex2, weight});
    this.adjacencyList[vertex2].push({node:vertex1, weight});
  }
  dijkstra(start, finish) {
    const nodes = new PriorityQueue();
    const distances = {};
    const previous = {};
    let smallest;
    let path = []; // to return at the end
    
    
    // build up initial state
    for(let vertex in this.adjacencyList) {
     if(vertex == start) {
       distances[vertex] = 0;
       nodes.enqueue(vertex, 0);
     } else {
       distances[vertex] = infinity;
       nodes.enqueue(vertex, Infinity);
     }
     previous[vertex] = null;
    }
    
    // as long as there is something to visit
    while(nodes.values.length) {
      smallest = nodes.dequeue().val;
      if(smallest == finish) {
      // we are done.
      // build up path to return at end
        while(previous[smallest]) {
          path.push(smallest);
          smallest = previous[smallest];
        }
        break;
      }
      if(smallest || distances[smallest] !== Infinity) {
        for(let neighbor in this.adjacencyList[smallest]) {
         // find neighboring node
         let nextNode = this.adjacencyList[smallest][neighbor];
         // calculate new distance to neighboring node
         let candidate = distances[smallest] + nextNode.weight;
         let nextNeighbor = nextNode.node;
         if(candidate < distances[nextNode.node]) {
           // updateing new smallest distance to neighbor
           distances[nextNeighbor] = candidate;
           // updating previous - how we got to neighbor
           previous[nextNeighbor] = smallest;
           // enqueue in priority queue with new priority
           nodes.enqueue(nextNeighbor, candidate);
         }
        }
      }
    }
    
    return path.concat(smallest).reverse();
  }
}

 

 

Using a binary heap priority queue is much faster than using a priority queue above.