激光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

相关文章

Java使用Javassist动态生成HelloWorld类

《Java使用Javassist动态生成HelloWorld类》Javassist是一个非常强大的字节码操作和定义库,它允许开发者在运行时创建新的类或者修改现有的类,本文将简单介绍如何使用Javass... 目录1. Javassist简介2. 环境准备3. 动态生成HelloWorld类3.1 创建CtC

SpringBoot 多环境开发实战(从配置、管理与控制)

《SpringBoot多环境开发实战(从配置、管理与控制)》本文详解SpringBoot多环境配置,涵盖单文件YAML、多文件模式、MavenProfile分组及激活策略,通过优先级控制灵活切换环境... 目录一、多环境开发基础(单文件 YAML 版)(一)配置原理与优势(二)实操示例二、多环境开发多文件版

Redis实现高效内存管理的示例代码

《Redis实现高效内存管理的示例代码》Redis内存管理是其核心功能之一,为了高效地利用内存,Redis采用了多种技术和策略,如优化的数据结构、内存分配策略、内存回收、数据压缩等,下面就来详细的介绍... 目录1. 内存分配策略jemalloc 的使用2. 数据压缩和编码ziplist示例代码3. 优化的

SpringBoot集成XXL-JOB实现任务管理全流程

《SpringBoot集成XXL-JOB实现任务管理全流程》XXL-JOB是一款轻量级分布式任务调度平台,功能丰富、界面简洁、易于扩展,本文介绍如何通过SpringBoot项目,使用RestTempl... 目录一、前言二、项目结构简述三、Maven 依赖四、Controller 代码详解五、Service

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象

Linux系统管理与进程任务管理方式

《Linux系统管理与进程任务管理方式》本文系统讲解Linux管理核心技能,涵盖引导流程、服务控制(Systemd与GRUB2)、进程管理(前台/后台运行、工具使用)、计划任务(at/cron)及常用... 目录引言一、linux系统引导过程与服务控制1.1 系统引导的五个关键阶段1.2 GRUB2的进化优

Spring Security 前后端分离场景下的会话并发管理

《SpringSecurity前后端分离场景下的会话并发管理》本文介绍了在前后端分离架构下实现SpringSecurity会话并发管理的问题,传统Web开发中只需简单配置sessionManage... 目录背景分析传统 web 开发中的 sessionManagement 入口ConcurrentSess

Python利用GeoPandas打造一个交互式中国地图选择器

《Python利用GeoPandas打造一个交互式中国地图选择器》在数据分析和可视化领域,地图是展示地理信息的强大工具,被将使用Python、wxPython和GeoPandas构建的交互式中国地图行... 目录技术栈概览代码结构分析1. __init__ 方法:初始化与状态管理2. init_ui 方法:

Linux之UDP和TCP报头管理方式

《Linux之UDP和TCP报头管理方式》文章系统讲解了传输层协议UDP与TCP的核心区别:UDP无连接、不可靠,适合实时传输(如视频),通过端口号标识应用;TCP有连接、可靠,通过确认应答、序号、窗... 目录一、关于端口号1.1 端口号的理解1.2 端口号范围的划分1.3 认识知名端口号1.4 一个进程

SpringBoot结合Knife4j进行API分组授权管理配置详解

《SpringBoot结合Knife4j进行API分组授权管理配置详解》在现代的微服务架构中,API文档和授权管理是不可或缺的一部分,本文将介绍如何在SpringBoot应用中集成Knife4j,并进... 目录环境准备配置 Swagger配置 Swagger OpenAPI自定义 Swagger UI 底