35 #define LOAD_MAP_FROM_FILE 0 // 1: load from "sMAP_FILE", 0: random map. 36 #define SHOW_POINT_LABELS 0 42 #if !LOAD_MAP_FROM_FILE 49 const std::string sMAP_FILE = string(
"./DLRMap.txt");
73 "MRPT example: ransac-data-association", 800, 600);
84 #if LOAD_MAP_FROM_FILE 90 const size_t nPts = M.
rows();
92 for (
size_t i = 0; i < nPts; i++) the_map.
setPoint(i, M(i, 1), M(i, 2));
105 const size_t nMapPts = the_map.
size();
106 cout <<
"Loaded/generated map with " << nMapPts <<
" landmarks.\n";
123 scene->getViewport(
"main")->setCustomBackgroundColor(
134 gl_map->loadFromPointsMap(&the_map);
135 gl_map->setColor(0, 0, 1);
136 gl_map->setPointSize(3);
138 scene->insert(gl_map);
140 #if SHOW_POINT_LABELS 141 for (
size_t i = 0; i < the_map.
size(); i++)
147 gl_txt->setLocation(x + 0.05, y + 0.05, 0.01);
149 scene->insert(gl_txt);
154 scene->insert(gl_lines);
157 gl_obs_map->setColor(1, 0, 0);
158 gl_obs_map->setPointSize(5);
160 gl_result->setColor(0, 1, 0);
161 gl_result->setPointSize(4);
165 gl_obs->insert(gl_obs_map);
166 gl_obs->insert(gl_obs_txts);
167 scene->insert(gl_obs);
168 scene->insert(gl_result);
170 win.unlockAccess3DScene();
178 vector<TObs> observations;
179 observations.resize(nObs);
190 std::vector<std::pair<size_t, float>> idxs;
194 for (
size_t i = 0; i < nObs; i++)
197 the_map.
getPoint(idxs[i].first, gx, gy);
202 observations[i].ID = idxs[i].first;
216 all_correspondences.reserve(nMapPts * nObs);
219 for (
size_t j = 0; j < nObs; j++)
224 match.
other_x = observations[j].x;
225 match.
other_y = observations[j].y;
227 for (
size_t i = 0; i < nMapPts; i++)
232 all_correspondences.push_back(match);
235 cout <<
"Generated " << all_correspondences.size()
236 <<
" potential pairings.\n";
241 timelog.
enter(
"robustRigidTransformation");
247 params.ransac_minSetSize =
250 params.ransac_maxSetSize =
253 params.ransac_mahalanobisDistanceThreshold =
255 params.ransac_nSimulations = 0;
256 params.ransac_fuseByCorrsMatch =
true;
257 params.ransac_fuseMaxDiffXY = 0.01f;
258 params.ransac_fuseMaxDiffPhi = 0.1_deg;
259 params.ransac_algorithmForLandmarks =
true;
260 params.probability_find_good_model = 0.999999;
261 params.ransac_min_nSimulations =
270 timelog.
leave(
"robustRigidTransformation");
275 const double tim = timer.
Tac();
279 cout <<
"# of SOG modes: " << best_poses.
size() << endl;
280 cout <<
"Best match has " << out_best_pairings.size() <<
" features:\n";
281 for (
size_t i = 0; i < out_best_pairings.size(); i++)
282 cout << out_best_pairings[i].this_idx <<
" <-> " 283 << out_best_pairings[i].other_idx << endl;
287 vector<int> obs2map_pairings(nObs, -1);
288 for (
size_t i = 0; i < out_best_pairings.size(); i++)
289 obs2map_pairings[out_best_pairings[i].other_idx] =
290 out_best_pairings[i].this_idx == ((
unsigned int)-1)
292 : out_best_pairings[i].this_idx;
295 for (
size_t i = 0; i < nObs; i++) cout << obs2map_pairings[i] <<
" ";
307 cout <<
"Solution pose: " << solution_pose.
mean << endl;
308 cout <<
"Ground truth pose: " << GT_pose << endl;
312 win.get3DSceneAndLock();
316 "Blue: map landmarks | Red: Observations | White lines: Found " 322 for (
size_t k = 0; k < nObs; k++)
323 gl_obs_map->insertPoint(
324 observations[k].x, observations[k].y, 0.0);
326 gl_obs->setPose(solution_pose.
mean);
328 #if SHOW_POINT_LABELS 329 gl_obs_txts->clear();
330 for (
size_t i = 0; i < nObs; i++)
334 const double x = observations[i].x;
335 const double y = observations[i].y;
336 gl_txt->setLocation(x + 0.05, y + 0.05, 0.01);
337 gl_obs_txts->insert(gl_txt);
345 for (
size_t k = 0; k < nObs; k++)
347 int map_idx = obs2map_pairings[k];
348 if (map_idx < 0)
continue;
352 the_map.
getPoint(map_idx, map_x, map_y);
356 observations[k].x, observations[k].y, obs_x, obs_y);
360 gl_lines->appendLine(map_x, map_y, 0, obs_x, obs_y, z);
362 sqerr += mrpt::math::distanceSqrBetweenPoints<double>(
363 map_x, map_y, obs_x, obs_y);
367 5, 20,
"Ground truth pose : " + GT_pose.asString(), 1);
370 "RANSAC estimated pose: " + solution_pose.
mean.
asString() +
371 mrpt::format(
" | RMSE=%f", (nPairs ? sqerr / nPairs : 0.0)),
374 win.unlockAccess3DScene();
377 cout <<
"nPairings: " << nPairs
378 <<
" RMSE = " << (nPairs ? sqerr / nPairs : 0.0) << endl;
396 catch (
const std::exception& e)
403 printf(
"Untyped exception!!");
A namespace of pseudo-random numbers generators of diferent distributions.
double Tac() noexcept
Stops the stopwatch.
void getPoint(size_t index, float &x, float &y, float &z) const
Access to a given point from map, as a 2D point.
CPose2D mean
The mean value.
const float normalizationStd
void asString(std::string &s) const
Returns a human-readable textual representation of the object (eg: "[x y yaw]", yaw in degrees) ...
static Ptr Create(Args &&... args)
Declares a class that represents a Probability Density function (PDF) of a 2D pose ...
std::string std::string format(std::string_view fmt, ARGS &&... args)
bool se2_l2(const mrpt::tfest::TMatchingPairList &in_correspondences, mrpt::math::TPose2D &out_transformation, mrpt::math::CMatrixDouble33 *out_estimateCovariance=nullptr)
Least-squares (L2 norm) solution to finding the optimal SE(2) (x,y,yaw) between two reference frames...
Parameters for se2_l2_robust().
void randomize(const uint32_t seed)
Initialize the PRNG from the given random seed.
const size_t MINIMUM_RANSAC_ITERS
A high-performance stopwatch, with typical resolution of nanoseconds.
const float ransac_mahalanobisDistanceThreshold
mrpt::vision::TStereoCalibParams params
A cloud of points in 2D or 3D, which can be built from a sequence of laser scans. ...
static Ptr Create(Args &&... args)
std::string formatTimeInterval(const double timeSeconds)
Returns a formated string with the given time difference (passed as the number of seconds)...
void composePoint(double lx, double ly, double &gx, double &gy) const
An alternative, slightly more efficient way of doing with G and L being 2D points and P this 2D pose...
mrpt::math::CMatrixDouble33 cov
The 3x3 covariance matrix.
static Ptr Create(Args &&... args)
#define ASSERT_(f)
Defines an assertion mechanism.
This base provides a set of functions for maths stuff.
size_t kdTreeRadiusSearch2D(const num_t x0, const num_t y0, const num_t maxRadiusSqr, std::vector< std::pair< size_t, num_t >> &out_indices_dist) const
KD Tree-based search for all the points within a given radius of some 2D point.
bool se2_l2_robust(const mrpt::tfest::TMatchingPairList &in_correspondences, const double in_normalizationStd, const TSE2RobustParams &in_ransac_params, TSE2RobustResult &out_results)
Robust least-squares (L2 norm) solution to finding the optimal SE(2) (x,y,yaw) between two reference ...
map< string, CVectorDouble > results
Declares a class that represents a Probability Density function (PDF) of a 2D pose ...
return_t drawGaussian1D(const double mean, const double std)
Generate a normally distributed pseudo-random number.
mrpt::gui::CDisplayWindow3D::Ptr win
size_type rows() const
Number of rows in the matrix.
size_type cols() const
Number of columns in the matrix.
A structure for holding correspondences between two sets of points or points-like entities in 2D or 3...
const size_t NUM_OBSERVATIONS_TO_SIMUL
return_t square(const num_t x)
Inline function for the square of a number.
void enter(const std::string_view &func_name) noexcept
Start of a named section.
This is the global namespace for all Mobile Robot Programming Toolkit (MRPT) libraries.
void setPoint(size_t index, float x, float y, float z)
Changes a given point from map, with Z defaulting to 0 if not provided.
A versatile "profiler" that logs the time spent within each pair of calls to enter(X)-leave(X), among other stats.
A class used to store a 2D pose, including the 2D coordinate point and a heading (phi) angle...
CSetOfObjects::Ptr CornerXYZ(float scale=1.0)
Returns three arrows representing a X,Y,Z 3D corner.
An RGBA color - floats in the range [0,1].
double leave(const std::string_view &func_name) noexcept
End of a named section.
std::string exception_to_str(const std::exception &e)
Builds a nice textual representation of a nested exception, which if generated using MRPT macros (THR...
size_t size() const
Return the number of Gaussian modes.
bool verbose
Show progress messages to std::cout console (default=true)
void Tic() noexcept
Starts the stopwatch.
CRandomGenerator & getRandomGenerator()
A static instance of a CRandomGenerator class, for use in single-thread applications.
size_t size() const
Save the point cloud as a PCL PCD file, in either ASCII or binary format.
void resize(size_t newLength) override
Resizes all point buffers so they can hold the given number of points: newly created points are set t...
Functions for estimating the optimal transformation between two frames of references given measuremen...
Output placeholder for se2_l2_robust()
const size_t NUM_MAP_FEATS
static Ptr Create(Args &&... args)
A graphical user interface (GUI) for efficiently rendering 3D scenes in real-time.
const size_t RANSAC_MINIMUM_INLIERS
void loadFromTextFile(std::istream &f)
Loads a vector/matrix from a text file, compatible with MATLAB text format.