MRPT  2.0.1
test.cpp
Go to the documentation of this file.
1 /* +------------------------------------------------------------------------+
2  | Mobile Robot Programming Toolkit (MRPT) |
3  | https://www.mrpt.org/ |
4  | |
5  | Copyright (c) 2005-2020, Individual contributors, see AUTHORS file |
6  | See: https://www.mrpt.org/Authors - All rights reserved. |
7  | Released under BSD License. See: https://www.mrpt.org/License |
8  +------------------------------------------------------------------------+ */
9 
11 #include <mrpt/graphs/dijkstra.h>
13 #include <mrpt/random.h>
14 #include <mrpt/system/CTicTac.h>
15 #include <iostream>
16 #include <map>
17 
18 using namespace mrpt;
19 using namespace mrpt::graphs;
20 using namespace mrpt::poses;
21 using namespace mrpt::math;
22 using namespace mrpt::gui;
23 using namespace mrpt::random;
24 using namespace mrpt::system;
25 using namespace std;
26 
27 // The type of my Dijkstra problem:
29  CMyDijkstra; // See other options in mrpt::poses::CNetworkOfPoses<>
30 
31 // Before MRPT 0.9.5 it was:
32 // typedef CDijkstra<CNetworkOfPoses2D::edge_t> CMyDijkstra;
33 
34 // adds a new edge to the graph. The edge is annotated with the relative
35 // position of the two nodes
36 void addEdge(
37  TNodeID from, TNodeID to, const std::map<TNodeID, CPose2D>& real_poses,
38  CNetworkOfPoses2D& graph_links)
39 {
40  CPose2D p = real_poses.find(to)->second - real_poses.find(from)->second;
41  graph_links.insertEdge(from, to, p);
42 }
43 
44 // weight is the distance between two nodes.
45 double myDijkstraWeight(
46  const CMyDijkstra::graph_t& g, const TNodeID from, const TNodeID to,
47  const CMyDijkstra::edge_t& edge)
48 {
49  // return 1; // Topological distance
50  return edge.norm(); // Metric distance
51 }
52 
53 // ------------------------------------------------------
54 // TestDijkstra
55 // ------------------------------------------------------
56 void TestDijkstra()
57 {
58  CTicTac tictac;
59  CNetworkOfPoses2D graph_links;
60  CNetworkOfPoses2D::global_poses_t optimal_poses, optimal_poses_dijkstra;
61  std::map<TNodeID, CPose2D> real_poses;
62 
64 
65  // ----------------------------
66  // Create a random graph:
67  // ----------------------------
68  const size_t N_VERTEX = 20;
69  const double DIST_THRES = 10;
70  const double NODES_XY_MAX = 15;
71 
72  vector<double> xs, ys;
73 
74  for (size_t j = 0; j < N_VERTEX; j++)
75  {
76  CPose2D p(
77  getRandomGenerator().drawUniform(-NODES_XY_MAX, NODES_XY_MAX),
78  getRandomGenerator().drawUniform(-NODES_XY_MAX, NODES_XY_MAX),
79  getRandomGenerator().drawUniform(-M_PI, M_PI));
80  real_poses[j] = p;
81 
82  // for the figure:
83  xs.push_back(p.x());
84  ys.push_back(p.y());
85  }
86 
87  // Add some edges
88  for (size_t i = 0; i < N_VERTEX; i++)
89  {
90  for (size_t j = 0; j < N_VERTEX; j++)
91  {
92  if (i == j) continue;
93  if (real_poses[i].distanceTo(real_poses[j]) < DIST_THRES)
94  addEdge(i, j, real_poses, graph_links);
95  }
96  }
97 
98  // ----------------------------
99  // Dijkstra
100  // ----------------------------
101  tictac.Tic();
102  const size_t SOURCE_NODE = 0;
103 
104  CMyDijkstra myDijkstra(graph_links, SOURCE_NODE, &myDijkstraWeight);
105 
106  cout << "Dijkstra took " << tictac.Tac() * 1e3 << " ms for "
107  << graph_links.edges.size() << " edges." << endl;
108 
109  // Demo of getting the tree representation of
110  // the graph & visit its nodes:
111  // ---------------------------------------------------------
112  CMyDijkstra::tree_graph_t graphAsTree;
113  myDijkstra.getTreeGraph(graphAsTree);
114 
115  // Text representation of the tree:
116  cout << "TREE:\n" << graphAsTree.getAsTextDescription() << endl;
117 
118  struct CMyVisitor : public CMyDijkstra::tree_graph_t::Visitor
119  {
120  void OnVisitNode(
121  const TNodeID parent,
122  const CMyDijkstra::tree_graph_t::TEdgeInfo& edge_to_child,
123  const size_t depth_level) override
124  {
125  cout << string(depth_level * 3, ' ');
126  cout << edge_to_child.id << endl;
127  }
128  };
129 
130  CMyVisitor myVisitor;
131 
132  cout << "Depth-first traverse of graph:\n";
133  cout << SOURCE_NODE << endl;
134  graphAsTree.visitDepthFirst(SOURCE_NODE, myVisitor);
135 
136  cout << endl << "Breadth-first traverse of graph:\n";
137  cout << SOURCE_NODE << endl;
138  graphAsTree.visitBreadthFirst(SOURCE_NODE, myVisitor);
139 
140  // ----------------------------
141  // Display results graphically:
142  // ----------------------------
143  CDisplayWindowPlots win("Dijkstra example");
144 
145  win.hold_on();
146  win.axis_equal();
147 
148  for (TNodeID i = 0; i < N_VERTEX && win.isOpen(); i++)
149  {
150  if (i == SOURCE_NODE) continue;
151 
153 
154  myDijkstra.getShortestPathTo(i, path);
155 
156  cout << "to " << i << " -> #steps= " << path.size() << endl;
157 
158  win.setWindowTitle(format(
159  "Dijkstra path %u->%u", static_cast<unsigned int>(SOURCE_NODE),
160  static_cast<unsigned int>(i)));
161 
162  win.clf();
163 
164  // plot all edges:
165  for (CNetworkOfPoses2D::iterator e = graph_links.begin();
166  e != graph_links.end(); ++e)
167  {
168  const CPose2D& p1 = real_poses[e->first.first];
169  const CPose2D& p2 = real_poses[e->first.second];
170 
171  vector<double> X(2);
172  vector<double> Y(2);
173  X[0] = p1.x();
174  Y[0] = p1.y();
175  X[1] = p2.x();
176  Y[1] = p2.y();
177  win.plot(X, Y, "k1");
178  }
179 
180  // Draw the shortest path:
181  for (CMyDijkstra::edge_list_t::const_iterator a = path.begin();
182  a != path.end(); ++a)
183  {
184  const CPose2D& p1 = real_poses[a->first];
185  const CPose2D& p2 = real_poses[a->second];
186 
187  vector<double> X(2);
188  vector<double> Y(2);
189  X[0] = p1.x();
190  Y[0] = p1.y();
191  X[1] = p2.x();
192  Y[1] = p2.y();
193  win.plot(X, Y, "g3");
194  }
195 
196  // Draw All nodes:
197  win.plot(xs, ys, ".b7");
198  win.axis_fit(true);
199 
200  cout << "Press any key to show next shortest path, close window to "
201  "end...\n";
202  win.waitForKey();
203  }
204 
205  win.clear();
206 }
207 
208 int main()
209 {
210  try
211  {
212  TestDijkstra();
213  return 0;
214  }
215  catch (exception& e)
216  {
217  cout << "MRPT exception caught: " << e.what() << endl;
218  return -1;
219  }
220  catch (...)
221  {
222  printf("Another exception!!");
223  return -1;
224  }
225 }
A namespace of pseudo-random numbers generators of diferent distributions.
CDijkstra< CNetworkOfPoses2D > CMyDijkstra
double Tac() noexcept
Stops the stopwatch.
Definition: CTicTac.cpp:86
A special kind of graph in the form of a tree with directed edges and optional edge annotations of te...
Definition: CDirectedTree.h:51
void addEdge(TNodeID from, TNodeID to, const std::map< TNodeID, CPose2D > &real_poses, CNetworkOfPoses2D &graph_links)
void TestDijkstra()
Create a GUI window and display plots with MATLAB-like interfaces and commands.
std::string std::string format(std::string_view fmt, ARGS &&... args)
Definition: format.h:26
edges_map_t edges
The public member with the directed edges in the graph.
typename graph_t::edge_t edge_t
The type of edge data in graph_t.
Definition: dijkstra.h:154
Abstract graph and tree data structures, plus generic graph algorithms.
void randomize(const uint32_t seed)
Initialize the PRNG from the given random seed.
A high-performance stopwatch, with typical resolution of nanoseconds.
STL namespace.
A directed graph of pose constraints, with edges being the relative poses between pairs of nodes iden...
std::list< TPairNodeIDs > edge_list_t
A list of edges used to describe a path on the graph.
Definition: dijkstra.h:156
This base provides a set of functions for maths stuff.
void visitDepthFirst(const TNodeID vroot, Visitor &user_visitor, const size_t root_depth_level=0) const
Depth-first visit of all children nodes of a given root (itself excluded from the visit)...
void insertEdge(TNodeID from_nodeID, TNodeID to_nodeID, const edge_t &edge_value)
Insert an edge (from -> to) with the given edge value.
typename MAPS_IMPLEMENTATION::template map< mrpt::graphs::TNodeID, global_pose_t > global_poses_t
A map from pose IDs to their global coordinate estimates, without uncertainty (the "most-likely value...
double x() const
Common members of all points & poses classes.
Definition: CPoseOrPoint.h:143
mrpt::gui::CDisplayWindow3D::Ptr win
Classes for 2D/3D geometry representation, both of single values and probability density distribution...
This is the global namespace for all Mobile Robot Programming Toolkit (MRPT) libraries.
TYPE_GRAPH graph_t
The type of the graph, typically a mrpt::graphs::CDirectedGraph<> or any other derived class...
Definition: dijkstra.h:152
A class used to store a 2D pose, including the 2D coordinate point and a heading (phi) angle...
Definition: CPose2D.h:39
uint64_t TNodeID
A generic numeric type for unique IDs of nodes or entities.
Definition: TNodeID.h:16
The Dijkstra algorithm for finding the shortest path between a given source node in a (weighted) dire...
Definition: dijkstra.h:97
Classes for creating GUI windows for 2D and 3D visualization.
Definition: about_box.h:14
void Tic() noexcept
Starts the stopwatch.
Definition: CTicTac.cpp:75
double myDijkstraWeight(const CMyDijkstra::graph_t &g, const TNodeID from, const TNodeID to, const CMyDijkstra::edge_t &edge)
CRandomGenerator & getRandomGenerator()
A static instance of a CRandomGenerator class, for use in single-thread applications.
std::string getAsTextDescription() const
Return a text representation of the tree spanned in a depth-first view, as in this example: ...
void visitBreadthFirst(const TNodeID vroot, Visitor &user_visitor, const size_t root_depth_level=0) const
Breadth-first visit of all children nodes of a given root (itself excluded from the visit)...



Page generated by Doxygen 1.8.14 for MRPT 2.0.1 Git: 0fef1a6d7 Fri Apr 3 23:00:21 2020 +0200 at vie abr 3 23:20:28 CEST 2020