激光SLAM如何动态管理关键帧和地图

2024-09-06 18:36

本文主要是介绍激光SLAM如何动态管理关键帧和地图,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

0. 简介

个人在想在长期执行的SLAM程序时,当场景发生替换时,激光SLAM如何有效的更新或者替换地图是非常关键的。在看了很多Life-Long的文章后,个人觉得可以按照以下思路去做。这里可以给大家分享一下

<br/>

1. 初始化保存关键帧

首先对应的应该是初始化设置,初始化设置当中会保存关键帧数据,这里的对应的关键帧点云数据会被存放在history_kf_lidar当中,这个数据是和关键帧状态一一对应的。

/*** @brief 初始化当前第一个关键帧*/
void dlio::OdomNode::initializeInputTarget() {// 构建第一个关键帧// 更新prev_scan_stampthis-&gt;prev_scan_stamp = this-&gt;scan_stamp;// 保存关键帧的姿态 以及去畸变降采样后的点云this-&gt;keyframes.push_back(std::make_pair(std::make_pair(this-&gt;lidarPose.p, this-&gt;lidarPose.q), this-&gt;current_scan));// 保存关键帧消息时间戳this-&gt;keyframe_timestamps.push_back(this-&gt;scan_header_stamp);// 保存关键帧点云协方差this-&gt;keyframe_normals.push_back(this-&gt;gicp.getSourceCovariances());// 初始T-corr为单位的 T = T_corr * T_priorthis-&gt;keyframe_transformations.push_back(this-&gt;T_corr);this-&gt;keyframe_transformations_prior.push_back(this-&gt;T_prior);this-&gt;keyframe_stateT.push_back(this-&gt;T);// 保存历史的lidar系点云std::unique_lock&lt;decltype(this-&gt;history_kf_lidar_mutex)&gt; lock_his_lidar(this-&gt;history_kf_lidar_mutex);pcl::PointCloud&lt;PointType&gt;::Ptr temp(new pcl::PointCloud&lt;PointType&gt;);pcl::copyPointCloud(*this-&gt;current_scan_lidar, *temp);this-&gt;history_kf_lidar.push_back(temp);lock_his_lidar.unlock();// 保存当前的关键帧信息 等待后端处理// tempKeyframe的姿态这里有两种处理方法// 1. 使用lidarPose 2. 使用state// 根据DLIO作者的解释 使用lidarPose更加稳定// (https://github.com/vectr-ucla/direct_lidar_inertial_odometry/issues/13#issuecomment-1638876779)this-&gt;currentFusionState = this-&gt;state;this-&gt;tempKeyframe.pos = this-&gt;lidarPose.p;this-&gt;tempKeyframe.rot = this-&gt;currentFusionState.q;this-&gt;tempKeyframe.vSim = {1};this-&gt;tempKeyframe.submap_kf_idx = {0};this-&gt;tempKeyframe.time = this-&gt;scan_stamp;this-&gt;v_kf_time.push_back(this-&gt;scan_stamp);pcl::copyPointCloud(*this-&gt;current_scan, *this-&gt;tempKeyframe.pCloud);this-&gt;KeyframesInfo.push_back(this-&gt;tempKeyframe);this-&gt;saveFirstKeyframeAndUpdateFactor();}();

在保存第一针状态并更新因子的时候,会同时缓存一个update_map_info。这个缓存了回环数据(依赖addLoopFactor函数)以及基于Gtsam计算出来的(iSAMCurrentEstimate)函数。

添加里程计因子 - 通过调用this->addOdomFactor(),函数向因子图中添加了一个里程计因子。里程计因子可能包含了从传感器得到的运动信息,这些信息用于估算机器人或设备的位置。

更新ISAM(增量平滑和映射) - 使用this->isam->update(this->gtSAMgraph, this->initialEstimate)来更新ISAM的状态,这个步骤通常包括增量优化,以便在获得新数据时快速更新地图或路径。然后再次调用this->isam->update()进行进一步的更新。 清空因子图和初始估计 - 将因子图和初始状态估计清空,这通常发生在更新完成之后,为下一轮更新做准备。 **计算当前估计 **- 使用this->isam->calculateEstimate()来计算当前状态的最佳估计。 获取临时关键帧信息 - 在互斥锁的保护下,从this->tempKeyframe获取临时关键帧的信息,并释放锁。 校正位置 - 通过调用this->correctPoses()函数来校正或调整已经估计的位置。 更新地图信息 - 处理是否发生了回环检测(由this->isLoop变量表示),并将当前估计状态和回环检测的结果推入更新队列this->update_map_info中,用于后续的地图更新或路径规划。

/*** @brief 对第一帧的处理*/
void dlio::OdomNode::saveFirstKeyframeAndUpdateFactor()
{// 添加里程计因子this-&gt;addOdomFactor();this-&gt;isam-&gt;update(this-&gt;gtSAMgraph, this-&gt;initialEstimate);this-&gt;isam-&gt;update();this-&gt;gtSAMgraph.resize(0);this-&gt;initialEstimate.clear();this-&gt;iSAMCurrentEstimate = this-&gt;isam-&gt;calculateEstimate();std::unique_lock&lt;decltype(this-&gt;tempKeyframe_mutex)&gt; lock_temp(this-&gt;tempKeyframe_mutex);KeyframeInfo his_info = this-&gt;tempKeyframe;lock_temp.unlock();this-&gt;correctPoses();auto loop_copy = this-&gt;isLoop;auto iSAMCurrentEstimate_copy  = this-&gt;iSAMCurrentEstimate;std::unique_lock&lt;decltype(this-&gt;update_map_info_mutex)&gt; lock_update_map(this-&gt;update_map_info_mutex);this-&gt;update_map_info.push(std::make_pair(loop_copy, iSAMCurrentEstimate_copy));lock_update_map.unlock();
}

在初始状态下,第一帧只用去考虑this->addOdomFactor();这个函数,用于添加odom的因子,里程计因子包含了从传感器得到的运动信息,这些信息用于估算机器人或设备的位置。

静态计数器- static int num_factor用作计数器,记录添加到因子图中的里程计因子数量。

获取当前关键帧信息 - 函数首先通过互斥锁保护的方式获取临时关键帧(this->tempKeyframe)的信息。

初始帧处理 - 如果是第一帧(num_factor为0),函数会设置一个非常小的噪声模型作为先验,表明对初始关键帧的位置非常有信心,并将其添加到因子图和初始估计中。

添加里程计因子 - 对于非第一帧的情况,函数会遍历当前关键帧匹配的子地图关键帧,并对每一个匹配计算一个从匹配关键帧到当前关键帧的位姿变换。这涉及到获取匹配关键帧的位姿并计算它们之间的相对变换(poseFrom.between(poseTo))。

计算噪声权重 - 函数会根据匹配的相似度计算噪声权重,相似度越高,噪声权重越小,表示对这个匹配的信心越大。这里依赖的是buildSubmapViaJaccard函数

添加因子到因子图 - 将计算得到的位姿变换和相应的噪声模型作为里程计因子添加到因子图中。

更新初始估计- 将当前关键帧的位姿添加到初始估计中。

因子计数器递增 - 最后,num_factor递增,为下一次调用准备

/*** @brief 添加里程计因子 包含连续里程计和非连续里程计*/
void dlio::OdomNode::addOdomFactor()
{static int num_factor = 0;std::unique_lock&lt;decltype(this-&gt;tempKeyframe_mutex)&gt; lock_temp(this-&gt;tempKeyframe_mutex);KeyframeInfo current_kf_info = this-&gt;tempKeyframe;lock_temp.unlock();// 初始第一帧if (num_factor == 0){gtsam::noiseModel::Diagonal::shared_ptr priorNoise = gtsam::noiseModel::Diagonal::Variances((gtsam::Vector(6) &lt;&lt; 1e-12, 1e-12, 1e-12, 1e-12, 1e-12, 1e-12).finished());this-&gt;gtSAMgraph.addPrior(0, state2Pose3(current_kf_info.rot, current_kf_info.pos), priorNoise);this-&gt;initialEstimate.insert(0, state2Pose3(current_kf_info.rot, current_kf_info.pos));}else{// 当前帧匹配的子地图关键帧索引std::vector&lt;int&gt; curr_submap_id = current_kf_info.submap_kf_idx;// 当前帧匹配的子地图关键帧相似度std::vector&lt;float&gt; curr_sim = current_kf_info.vSim;// 当前帧的位姿gtsam::Pose3 poseTo = state2Pose3(current_kf_info.rot, current_kf_info.pos);// 遍历与当前帧匹配的子地图关键帧for (int i = 0; i &lt; curr_submap_id.size(); i++){// 子地图关键帧的位姿std::unique_lock&lt;decltype(this-&gt;keyframes_mutex)&gt; lock_kf(this-&gt;keyframes_mutex);gtsam::Pose3 poseFrom = state2Pose3(this-&gt;KeyframesInfo[curr_submap_id[i]].rot,this-&gt;KeyframesInfo[curr_submap_id[i]].pos);lock_kf.unlock();// 噪声权重double weight = 1.0;if (curr_sim.size() &gt; 0){// 对相似度进行归一化auto max_sim = std::max_element(curr_sim.begin(), curr_sim.end());for (auto &amp;it: curr_sim)it = it / *max_sim;double sim = curr_sim[i] &gt;= 1 ? 0.99 : curr_sim[i] &lt; 0 ? 0 : curr_sim[i];// 相似度越大 噪声权重越小weight = (1 - sim) * this-&gt;icpScore;}// 添加相邻/不相邻里程计因子gtsam::noiseModel::Diagonal::shared_ptr odometryNoise = gtsam::noiseModel::Diagonal::Variances((gtsam::Vector(6) &lt;&lt; 1e-6, 1e-6, 1e-6, 1e-4, 1e-4, 1e-4).finished() * weight);this-&gt;gtSAMgraph.emplace_shared&lt;gtsam::BetweenFactor&lt;gtsam::Pose3&gt;&gt;(curr_submap_id[i],num_factor,poseFrom.between(poseTo), odometryNoise);}this-&gt;initialEstimate.insert(num_factor, poseTo);}num_factor++;}

点击激光SLAM如何动态管理关键帧和地图——古月居可查看全文

这篇关于激光SLAM如何动态管理关键帧和地图的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1142801

相关文章

Spring Boot集成Druid实现数据源管理与监控的详细步骤

《SpringBoot集成Druid实现数据源管理与监控的详细步骤》本文介绍如何在SpringBoot项目中集成Druid数据库连接池,包括环境搭建、Maven依赖配置、SpringBoot配置文件... 目录1. 引言1.1 环境准备1.2 Druid介绍2. 配置Druid连接池3. 查看Druid监控

一文详解SpringBoot中控制器的动态注册与卸载

《一文详解SpringBoot中控制器的动态注册与卸载》在项目开发中,通过动态注册和卸载控制器功能,可以根据业务场景和项目需要实现功能的动态增加、删除,提高系统的灵活性和可扩展性,下面我们就来看看Sp... 目录项目结构1. 创建 Spring Boot 启动类2. 创建一个测试控制器3. 创建动态控制器注

Knife4j+Axios+Redis前后端分离架构下的 API 管理与会话方案(最新推荐)

《Knife4j+Axios+Redis前后端分离架构下的API管理与会话方案(最新推荐)》本文主要介绍了Swagger与Knife4j的配置要点、前后端对接方法以及分布式Session实现原理,... 目录一、Swagger 与 Knife4j 的深度理解及配置要点Knife4j 配置关键要点1.Spri

springboot如何通过http动态操作xxl-job任务

《springboot如何通过http动态操作xxl-job任务》:本文主要介绍springboot如何通过http动态操作xxl-job任务的问题,具有很好的参考价值,希望对大家有所帮助,如有错... 目录springboot通过http动态操作xxl-job任务一、maven依赖二、配置文件三、xxl-

基于Linux的ffmpeg python的关键帧抽取

《基于Linux的ffmpegpython的关键帧抽取》本文主要介绍了基于Linux的ffmpegpython的关键帧抽取,实现以按帧或时间间隔抽取关键帧,文中通过示例代码介绍的非常详细,对大家的学... 目录1.FFmpeg的环境配置1) 创建一个虚拟环境envjavascript2) ffmpeg-py

springboot项目中整合高德地图的实践

《springboot项目中整合高德地图的实践》:本文主要介绍springboot项目中整合高德地图的实践,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一:高德开放平台的使用二:创建数据库(我是用的是mysql)三:Springboot所需的依赖(根据你的需求再

使用jenv工具管理多个JDK版本的方法步骤

《使用jenv工具管理多个JDK版本的方法步骤》jenv是一个开源的Java环境管理工具,旨在帮助开发者在同一台机器上轻松管理和切换多个Java版本,:本文主要介绍使用jenv工具管理多个JD... 目录一、jenv到底是干啥的?二、jenv的核心功能(一)管理多个Java版本(二)支持插件扩展(三)环境隔

Java调用C#动态库的三种方法详解

《Java调用C#动态库的三种方法详解》在这个多语言编程的时代,Java和C#就像两位才华横溢的舞者,各自在不同的舞台上展现着独特的魅力,然而,当它们携手合作时,又会碰撞出怎样绚丽的火花呢?今天,我们... 目录方法1:C++/CLI搭建桥梁——Java ↔ C# 的“翻译官”步骤1:创建C#类库(.NET

Python中bisect_left 函数实现高效插入与有序列表管理

《Python中bisect_left函数实现高效插入与有序列表管理》Python的bisect_left函数通过二分查找高效定位有序列表插入位置,与bisect_right的区别在于处理重复元素时... 目录一、bisect_left 基本介绍1.1 函数定义1.2 核心功能二、bisect_left 与

MyBatis编写嵌套子查询的动态SQL实践详解

《MyBatis编写嵌套子查询的动态SQL实践详解》在Java生态中,MyBatis作为一款优秀的ORM框架,广泛应用于数据库操作,本文将深入探讨如何在MyBatis中编写嵌套子查询的动态SQL,并结... 目录一、Myhttp://www.chinasem.cnBATis动态SQL的核心优势1. 灵活性与可