Graph Algorithm Series: Shortest Path in Computational Graph

Graph Algorithm Series: Shortest Path in Computational Graph

[[398324]]

This article is reprinted from the WeChat public account "Beta Learns JAVA", the author is Silently9527. Please contact the Beta Learns JAVA public account to reprint this article.

Preface

In the previous two articles, we used depth-first search to find a path from vertex v to vertex w in the graph. However, depth-first search has a lot to do with vertex input, and the path found is not necessarily the shortest. Usually, we often need to find the shortest path in the graph, such as map function. Here we need to use the breadth-first search algorithm.

Breadth-First Search

Still use the previously defined path finding API

  1. public class Paths {
  2. Paths(Graph graph, int s);
  3.      
  4. boolean hasPathTo( int v); //Determine whether there is a path from s->v
  5.      
  6. Iterable< Integer > pathTo( int v); //If the path exists, return the path
  7. }

In breadth-first search, in order to find the shortest path, we need to traverse all vertices in the order of the starting point, instead of using recursion to achieve it; the idea of ​​the algorithm is:

  • Use a queue to store vertices that have been marked but whose adjacency lists have not yet been traversed.
  • Take the next vertex v from the queue and mark it
  • Add all unmarked vertices adjacent to v to the queue

In this algorithm, in order to save the path, we still need to use an edge array edgeTo[] and use a parent link tree to represent the shortest path from the root node to all connected vertices.

  1. public class BreadthFirstPaths {
  2. private boolean marked[];
  3. private int [] edgeTo;
  4. private int s;
  5. private Queue< Integer > queue = new LinkedListQueue<>();
  6.  
  7. public BreadthFirstPaths(Graph graph, int s) {
  8. this.s = s;
  9. this.marked = new boolean[graph.V()];
  10. this.edgeTo = new int [graph.V()];
  11.  
  12. bfs(graph, s);
  13. }
  14.  
  15. private void bfs(Graph graph, int s) {
  16. this.marked[s] = true ;
  17. this.queue.enqueue(s);
  18. while (!this.queue.isEmpty()) {
  19. Integer v = this.queue.dequeue();
  20. for ( int w : graph.adj(v)) {
  21. if (!this.marked[w]) {
  22. this.marked[w] = true ;
  23. this.edgeTo[w] = v;
  24. this.queue.enqueue(w);
  25. }
  26. }
  27. }
  28.  
  29.  
  30. }
  31.  
  32. public boolean hasPathTo( int v) {
  33. return this.marked[v];
  34. }
  35.  
  36. public Iterable< Integer > pathTo( int v) {
  37. if (!hasPathTo(v)) {
  38. throw new IllegalStateException( "s cannot reach v" );
  39. }
  40. Stack< Integer > stack = new LinkedListStack<>();
  41. stack.push(v);
  42. while (edgeTo[v] != s) {
  43. stack.push(edgeTo[v]);
  44. v = edgeTo[v];
  45. }
  46. stack.push(s);
  47. return stack;
  48. }
  49. }

Take the following figure as a column to see the running track of breadth-first search

Unit test code:

  1. @Test
  2. public void test() {
  3. Graph graph = new Graph(8);
  4. graph.addEdge(0, 1);
  5. graph.addEdge(0, 2);
  6. graph.addEdge(0, 5);
  7. graph.addEdge(1, 3);
  8. graph.addEdge(2, 4);
  9. graph.addEdge(4, 3);
  10. graph.addEdge(5, 3);
  11. graph.addEdge(6, 7);
  12.  
  13. BreadthFirstPaths paths = new BreadthFirstPaths(graph, 0);
  14. System. out .println(paths.hasPathTo(5));
  15. System. out .println(paths.hasPathTo(2));
  16. System. out .println(paths.hasPathTo(6));
  17.  
  18. paths.pathTo(5).forEach(System. out ::print);
  19. System.out.println( ) ;
  20. paths.pathTo(4).forEach(System. out ::print);
  21. System.out.println( ) ;
  22. paths.pathTo(2).forEach(System. out ::print);
  23. System.out.println( ) ;
  24. paths.pathTo(3).forEach(System. out ::print);
  25.  
  26. }

The execution results are consistent with the running trajectory we analyzed

Symbol diagram

The graph algorithms that we have learned in recent articles all use numbers as vertices because it is very simple and convenient to implement these algorithms with numbers. However, in actual scenarios, characters are usually used as vertices instead of numbers, such as: locations on a map, and the relationship between movies and actors.

In order to meet the actual scenario, we only need to make a mapping between numbers and strings. At this time, we will think of the map implemented in the previous article (map is implemented through binary tree, red-black tree, hash table, etc., interested students can check it out), and use Map to maintain the mapping relationship between strings and numbers.

  1. public interface SymbolGraph {
  2. boolean contains (String key ); //Judge whether a vertex exists
  3.  
  4. int   index (String key ); //Return the corresponding digital vertex by name
  5.  
  6. String name ( int v); //Return the corresponding character name through the digital vertex
  7.  
  8. Graph graph();
  9. }

Implementation ideas:

  • Use Map to save string-number mapping, key is string, value is number
  • Use an array to reverse the mapping from numbers to strings, where the array subscripts correspond to the numerical vertices and the values ​​correspond to the string names
  • Assume that the vertices and edges of the constructed graph are represented by strings, such as: a,b,c,d\nb,a,h,l,p\ng,s,z. The first string of each segment separated by \n represents the vertex v, and the following represents the adjacent vertices connected to the vertex v.

The actual process can be determined according to the specific situation. It does not necessarily have to be this kind of string. It can come from a database or a network request.

The code is implemented as follows:

  1. public class SymbolGraph {
  2. private Map<String, Integer > map = new RedBlack23RedBlackTreeMap<>();
  3. private String[] keys;
  4. private Graph graph;
  5.  
  6. public SymbolGraph(String text) {
  7. Arrays.stream(text.split( "\n" )).forEach(line -> {
  8. String[] split = line.split( "," );
  9. for ( int i = 0; i < split.length; i++) {
  10. map.put(split[i], i);
  11. }
  12. });
  13.  
  14. this.keys = new String[map. size ()];
  15. map.keys().forEach( key -> {
  16. this.keys[this.map.get( key )] = key ;
  17. });
  18.  
  19. this.graph = new Graph(this.keys.length);
  20. Arrays.stream(text.split( "\n" )).forEach(line -> {
  21. String[] split = line.split( "," );
  22. int v = this.map.get(split[0]);
  23. for ( int i = 1; i < split.length; i++) {
  24. this.graph.addEdge(v, this.map.get(split[i]));
  25. }
  26. });
  27.          
  28. }
  29.  
  30. public boolean contains (String key ) {
  31. return map.contains ( key ) ;
  32. }
  33.  
  34. public   int   index (String key ) {
  35. return map.get( key );
  36. }
  37.  
  38. public String name ( int   index ) {
  39. return this.keys[ index ];
  40. }
  41.  
  42. public Graph graph() {
  43. return this.graph;
  44. }
  45.  
  46. public   static void main(String[] args) {
  47. System. out .println(Arrays.toString( "323\n2323" .split( "\n" )));
  48. }
  49. }

All source code in this article has been put into the github repository: https://github.com/silently9527/JavaCore

<<:  5 Common SD-WAN Challenges and How to Address Them

>>:  How to determine whether the protocol is Websocket in Http Header

Recommend

From telegraph to 5G, 7 changes in the 60-year history of mobile communications

The history of communication has been around for ...

Talk about the past and present of WiFi

When it comes to the development of WiFi, we have...

Have you learned how to configure multiple public IP addresses?

background For some customers working on video an...

edgeNAT: 20% off monthly VPS and 30% off annual VPS, free CDN when you buy VPS

edgeNAT has launched a new CDN product this month...

Breaking news! 5G standard postponed for 3 months

According to reliable intelligence, at the 3GPP m...

Zigbee vs. Wi-Fi: Which is Better for Your Smart Home?

All smart home appliances rely on connectivity to...

Let 5G play a role earlier and make 5G technology 4G

The popularity and application of 4G has opened t...

Will 5G be the connectivity miracle for the Internet of Things?

Thanks to 5G, high-speed internet will soon be av...

In the new era, how can operators seize the opportunity of industrial Internet?

2020 is the year when 5G enters large-scale appli...