242 lines
6.3 KiB
V
242 lines
6.3 KiB
V
|
/*
|
||
|
Exploring Dijkstra,
|
||
|
The data example is from
|
||
|
https://www.geeksforgeeks.org/dijkstras-shortest-path-algorithm-greedy-algo-7/
|
||
|
|
||
|
by CCS
|
||
|
Dijkstra's single source shortest path algorithm.
|
||
|
The program uses an adjacency matrix representation of a graph
|
||
|
|
||
|
This Dijkstra algorithm uses a priority queue to save
|
||
|
the shortest paths. The queue structure has a data
|
||
|
which is the number of the node,
|
||
|
and the priority field which is the shortest distance.
|
||
|
|
||
|
PS: all the pre-requisites of Dijkstra are considered
|
||
|
|
||
|
$ v run file_name.v
|
||
|
// Creating a executable
|
||
|
$ v run file_name.v -o an_executable.EXE
|
||
|
$ ./an_executable.EXE
|
||
|
|
||
|
Code based from : Data Structures and Algorithms Made Easy: Data Structures and Algorithmic Puzzles, Fifth Edition (English Edition)
|
||
|
pseudo code written in C
|
||
|
This idea is quite different: it uses a priority queue to store the current
|
||
|
shortest path evaluted
|
||
|
The priority queue structure built using a list to simulate
|
||
|
the queue. A heap is not used in this case.
|
||
|
*/
|
||
|
|
||
|
// a structure
|
||
|
struct NODE {
|
||
|
mut:
|
||
|
data int // NUMBER OF NODE
|
||
|
priority int // Lower values priority indicate ==> higher priority
|
||
|
}
|
||
|
|
||
|
// Function to push according to priority ... the lower priority is goes ahead
|
||
|
// The "push" always sorted in pq
|
||
|
fn push_pq<T>(mut prior_queue []T, data int, priority int) {
|
||
|
mut temp := []T{}
|
||
|
lenght_pq := prior_queue.len
|
||
|
|
||
|
mut i := 0
|
||
|
for (i < lenght_pq) && (priority > prior_queue[i].priority) {
|
||
|
temp << prior_queue[i]
|
||
|
i++
|
||
|
}
|
||
|
// INSERTING SORTED in the queue
|
||
|
temp << NODE{data, priority} // do the copy in the right place
|
||
|
// copy the another part (tail) of original prior_queue
|
||
|
for i < lenght_pq {
|
||
|
temp << prior_queue[i]
|
||
|
i++
|
||
|
}
|
||
|
prior_queue = temp.clone() // I am not sure if it the right way
|
||
|
// IS IT THE RIGHT WAY?
|
||
|
}
|
||
|
|
||
|
// Change the priority of a value/node ... exist a value, change its priority
|
||
|
fn updating_priority<T>(mut prior_queue []T, search_data int, new_priority int) {
|
||
|
mut i := 0
|
||
|
mut lenght_pq := prior_queue.len
|
||
|
|
||
|
for i < lenght_pq {
|
||
|
if search_data == prior_queue[i].data {
|
||
|
prior_queue[i] = NODE{search_data, new_priority} // do the copy in the right place
|
||
|
break
|
||
|
}
|
||
|
i++
|
||
|
// all the list was examined
|
||
|
if i >= lenght_pq {
|
||
|
print('\n This data $search_data does exist ... PRIORITY QUEUE problem\n')
|
||
|
exit(1) // panic(s string)
|
||
|
}
|
||
|
} // end for
|
||
|
}
|
||
|
|
||
|
// a single departure or remove from queue
|
||
|
fn departure_priority<T>(mut prior_queue []T) int {
|
||
|
mut x := prior_queue[0].data
|
||
|
prior_queue.delete(0) // or .delete_many(0, 1 )
|
||
|
return x
|
||
|
}
|
||
|
|
||
|
// give a NODE v, return a list with all adjacents
|
||
|
// Take care, only positive EDGES
|
||
|
fn all_adjacents<T>(g [][]T, v int) []int {
|
||
|
mut temp := []int{} //
|
||
|
for i in 0 .. (g.len) {
|
||
|
if g[v][i] > 0 {
|
||
|
temp << i
|
||
|
}
|
||
|
}
|
||
|
return temp
|
||
|
}
|
||
|
|
||
|
// print the costs from origin up to all nodes
|
||
|
fn print_solution<T>(dist []T) {
|
||
|
print('Vertex \tDistance from Source')
|
||
|
for node in 0 .. (dist.len) {
|
||
|
print('\n $node ==> \t ${dist[node]}')
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// print all paths and their cost or weight
|
||
|
fn print_paths_dist<T>(path []T, dist []T) {
|
||
|
print('\n Read the nodes from right to left (a path): \n')
|
||
|
|
||
|
for node in 1 .. (path.len) {
|
||
|
print('\n $node ')
|
||
|
mut i := node
|
||
|
for path[i] != -1 {
|
||
|
print(' <= ${path[i]} ')
|
||
|
i = path[i]
|
||
|
}
|
||
|
print('\t PATH COST: ${dist[node]}')
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// check structure from: https://www.geeksforgeeks.org/dijkstras-shortest-path-algorithm-greedy-algo-7/
|
||
|
// s: source for all nodes
|
||
|
// Two results are obtained ... cost and paths
|
||
|
fn dijkstra(g [][]int, s int) {
|
||
|
mut pq_queue := []NODE{} // creating a priority queue
|
||
|
push_pq(mut pq_queue, s, 0) // goes s with priority 0
|
||
|
mut n := g.len
|
||
|
|
||
|
mut dist := []int{len: n, init: -1} // dist with -1 instead of INIFINITY
|
||
|
mut path := []int{len: n, init: -1} // previous node of each shortest paht
|
||
|
|
||
|
// Distance of source vertex from itself is always 0
|
||
|
dist[s] = 0
|
||
|
|
||
|
for pq_queue.len != 0 {
|
||
|
mut v := departure_priority(mut pq_queue)
|
||
|
// for all W adjcents vertices of v
|
||
|
mut adjs_of_v := all_adjacents(g, v) // all_ADJ of v ....
|
||
|
// print('\n ADJ ${v} is ${adjs_of_v}')
|
||
|
mut new_dist := 0
|
||
|
for w in adjs_of_v {
|
||
|
new_dist = dist[v] + g[v][w]
|
||
|
if dist[w] == -1 {
|
||
|
dist[w] = new_dist
|
||
|
push_pq(mut pq_queue, w, dist[w])
|
||
|
path[w] = v // collecting the previous node -- lowest weight
|
||
|
}
|
||
|
if dist[w] > new_dist {
|
||
|
dist[w] = new_dist
|
||
|
updating_priority(mut pq_queue, w, dist[w])
|
||
|
path[w] = v //
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// print the constructed distance array
|
||
|
print_solution(dist)
|
||
|
// print('\n \n Previous node of shortest path: ${path}')
|
||
|
print_paths_dist(path, dist)
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Solution Expected
|
||
|
Vertex Distance from Source
|
||
|
0 0
|
||
|
1 4
|
||
|
2 12
|
||
|
3 19
|
||
|
4 21
|
||
|
5 11
|
||
|
6 9
|
||
|
7 8
|
||
|
8 14
|
||
|
*/
|
||
|
|
||
|
fn main() {
|
||
|
// adjacency matrix = cost or weight
|
||
|
graph_01 := [
|
||
|
[0, 4, 0, 0, 0, 0, 0, 8, 0],
|
||
|
[4, 0, 8, 0, 0, 0, 0, 11, 0],
|
||
|
[0, 8, 0, 7, 0, 4, 0, 0, 2],
|
||
|
[0, 0, 7, 0, 9, 14, 0, 0, 0],
|
||
|
[0, 0, 0, 9, 0, 10, 0, 0, 0],
|
||
|
[0, 0, 4, 14, 10, 0, 2, 0, 0],
|
||
|
[0, 0, 0, 0, 0, 2, 0, 1, 6],
|
||
|
[8, 11, 0, 0, 0, 0, 1, 0, 7],
|
||
|
[0, 0, 2, 0, 0, 0, 6, 7, 0],
|
||
|
]
|
||
|
|
||
|
graph_02 := [
|
||
|
[0, 2, 0, 6, 0],
|
||
|
[2, 0, 3, 8, 5],
|
||
|
[0, 3, 0, 0, 7],
|
||
|
[6, 8, 0, 0, 9],
|
||
|
[0, 5, 7, 9, 0],
|
||
|
]
|
||
|
// data from https://www.geeksforgeeks.org/prims-minimum-spanning-tree-mst-greedy-algo-5/
|
||
|
/*
|
||
|
The graph:
|
||
|
2 3
|
||
|
(0)--(1)--(2)
|
||
|
| / \ |
|
||
|
6| 8/ \5 |7
|
||
|
| / \ |
|
||
|
(3)-------(4)
|
||
|
9
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
Let us create following weighted graph
|
||
|
From https://www.geeksforgeeks.org/kruskals-minimum-spanning-tree-algorithm-greedy-algo-2/?ref=lbp
|
||
|
10
|
||
|
0--------1
|
||
|
| \ |
|
||
|
6| 5\ |15
|
||
|
| \ |
|
||
|
2--------3
|
||
|
4
|
||
|
*/
|
||
|
graph_03 := [
|
||
|
[0, 10, 6, 5],
|
||
|
[10, 0, 0, 15],
|
||
|
[6, 0, 0, 4],
|
||
|
[5, 15, 4, 0],
|
||
|
]
|
||
|
|
||
|
// To find number of coluns
|
||
|
// mut cols := an_array[0].len
|
||
|
mut graph := [][]int{} // the graph: adjacency matrix
|
||
|
// for index, g_value in [graph_01, graph_02, graph_03] {
|
||
|
for index, g_value in [graph_01, graph_02, graph_03] {
|
||
|
graph = g_value.clone() // graphs_sample[g].clone() // choice your SAMPLE
|
||
|
// allways starting by node 0
|
||
|
start_node := 0
|
||
|
println('\n\n Graph ${index + 1} using Dijkstra algorithm (source node: $start_node)')
|
||
|
dijkstra(graph, start_node)
|
||
|
}
|
||
|
|
||
|
println('\n BYE -- OK')
|
||
|
}
|
||
|
|
||
|
//********************************************************************
|