Small knowledge, big challenge! This article is participating in the creation activity of “Essential Tips for Programmers”.

Talk about using Dijkstra to find the shortest path to a digraph

0 x00 preface

We all know that there are many ways to find the shortest path, such as Dijkstra algorithm, Bellman-Ford algorithm, Floyd-Warshall algorithm and so on. These algorithms have their own advantages and disadvantages. Among them, Floyd-Warshall algorithm has high time complexity, but low coding complexity. Bellman-ford algorithm is suitable for the case with negative edge weights. As for the Dijkstra algorithm discussed in this paper, the advantage is that the time complexity is small, but it cannot deal with graphs with negative weight edges.

We need to choose different algorithms for different situations.

0x01 Code analysis

#include<stdio.h>
#define MAXVEX 10
int graph[MAXVEX][MAXVEX];
int short_path[MAXVEX];
int prev_v[MAXVEX];
Copy the code

To import the standard I/O library, we set the number of nodes to 10, then set graph to store the graph, short_path to store the length of the shortest path from the source point to other points, and prev_v to store the pre-node of each node on the shortest path.

1. The figure

void create_graph( void ){ for(int i=0; i<10; i++){ for(int j=0; j<10; j++){ if(i==j){ graph[i][j]=0; continue; } graph[i][j] = 99999; }} printf(" Please enter the number of directed edges :"); int n; scanf("%d",&n); for(int k=0; k<n; k++){ int i,j,value; scanf("%d %d %d",&i,&j,&value); graph[i][j]=value; }}Copy the code

Then it’s time to initialize the graph. Here we initialize the distance between each vertex and ourselves to 0, then initialize all the other points to 99999, and then assign weights to each edge.

2. Find the shortest path

void Dijkstra(int g[10][10], int v0) { int final[MAXVEX]; int j = 0; int k = 0; for(j = 0; j<MAXVEX; j++) { final[j] = 0; short_path[j] = g[v0][j]; prev_v[j] = 0; } final[v0] = 1; short_path[v0] = 0; for(int i = 1; i < MAXVEX; i++) { int min = MAXVEX; k = 0; for(int j=0; j<MAXVEX; j++) { if(final[j]==0 && short_path[j]<min) { min = short_path[j]; k = j; } } final[k] = 1; for(int j=0; j<MAXVEX; j++) { if(final[j]==0 && min+g[k][j] < short_path[j]) { short_path[j] = min+g[k][j]; prev_v[j] = k; }}}}Copy the code

Here comes the feast! Here comes the Dijkstra algorithm!

The parameters passed to this function are the graph and the source point.

We use an array to mark whether the shortest path has been found from the source point to that point.

Initialize final, short_PATH, and prev_v, and add all edges of v0 to short_path.

Since the shortest distance from source to source is already determined to be 0, we mark final[v0] as 1 and short_path[v0] as 0.

Then the first loop that follows is to find maxvex-1 shortest paths and find the shortest path from the source point to the remaining unmarked vertices. The next point found is then marked to indicate that the path from the source point to that point has been found. Then look at the shortest path from the source point to other vertices (the vertex of the shortest path has not been determined), if the length of the path through vertex K is shorter than the original path, update. And set k as the leading node of J.

3. Output the shortest path

void show_shortest_path( int source ,int end) { int que[10]; int tot = 1; Printf (" shortest path to %d: ",end); que[tot] = end; tot++; int i = prev_v[end]; while(i ! = source) { que[tot] = i; tot++; i = prev_v[i]; } que[tot] = source; for(int i=tot; i>=1; i--){ if(i! =1){ printf("%d->",que[i]); }else{ printf("%d",que[i]); } } printf("\n"); }Copy the code

You build an array as a queue, and then you put each node you find in the queue, and then you walk through it sequentially.

\