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

相关文章

无人叉车3d激光slam多房间建图定位异常处理方案-墙体画线地图切分方案

墙体画线地图切分方案 针对问题:墙体两侧特征混淆误匹配,导致建图和定位偏差,表现为过门跳变、外月台走歪等 ·解决思路:预期的根治方案IGICP需要较长时间完成上线,先使用切分地图的工程化方案,即墙体两侧切分为不同地图,在某一侧只使用该侧地图进行定位 方案思路 切分原理:切分地图基于关键帧位置,而非点云。 理论基础:光照是直线的,一帧点云必定只能照射到墙的一侧,无法同时照到两侧实践考虑:关

第10章 中断和动态时钟显示

第10章 中断和动态时钟显示 从本章开始,按照书籍的划分,第10章开始就进入保护模式(Protected Mode)部分了,感觉从这里开始难度突然就增加了。 书中介绍了为什么有中断(Interrupt)的设计,中断的几种方式:外部硬件中断、内部中断和软中断。通过中断做了一个会走的时钟和屏幕上输入字符的程序。 我自己理解中断的一些作用: 为了更好的利用处理器的性能。协同快速和慢速设备一起工作

综合安防管理平台LntonAIServer视频监控汇聚抖动检测算法优势

LntonAIServer视频质量诊断功能中的抖动检测是一个专门针对视频稳定性进行分析的功能。抖动通常是指视频帧之间的不必要运动,这种运动可能是由于摄像机的移动、传输中的错误或编解码问题导致的。抖动检测对于确保视频内容的平滑性和观看体验至关重要。 优势 1. 提高图像质量 - 清晰度提升:减少抖动,提高图像的清晰度和细节表现力,使得监控画面更加真实可信。 - 细节增强:在低光条件下,抖

动态规划---打家劫舍

题目: 你是一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。 给定一个代表每个房屋存放金额的非负整数数组,计算你 不触动警报装置的情况下 ,一夜之内能够偷窃到的最高金额。 思路: 动态规划五部曲: 1.确定dp数组及含义 dp数组是一维数组,dp[i]代表

软考系统规划与管理师考试证书含金量高吗?

2024年软考系统规划与管理师考试报名时间节点: 报名时间:2024年上半年软考将于3月中旬陆续开始报名 考试时间:上半年5月25日到28日,下半年11月9日到12日 分数线:所有科目成绩均须达到45分以上(包括45分)方可通过考试 成绩查询:可在“中国计算机技术职业资格网”上查询软考成绩 出成绩时间:预计在11月左右 证书领取时间:一般在考试成绩公布后3~4个月,各地领取时间有所不同

安全管理体系化的智慧油站开源了。

AI视频监控平台简介 AI视频监控平台是一款功能强大且简单易用的实时算法视频监控系统。它的愿景是最底层打通各大芯片厂商相互间的壁垒,省去繁琐重复的适配流程,实现芯片、算法、应用的全流程组合,从而大大减少企业级应用约95%的开发成本。用户只需在界面上进行简单的操作,就可以实现全视频的接入及布控。摄像头管理模块用于多种终端设备、智能设备的接入及管理。平台支持包括摄像头等终端感知设备接入,为整个平台提

代码随想录冲冲冲 Day39 动态规划Part7

198. 打家劫舍 dp数组的意义是在第i位的时候偷的最大钱数是多少 如果nums的size为0 总价值当然就是0 如果nums的size为1 总价值是nums[0] 遍历顺序就是从小到大遍历 之后是递推公式 对于dp[i]的最大价值来说有两种可能 1.偷第i个 那么最大价值就是dp[i-2]+nums[i] 2.不偷第i个 那么价值就是dp[i-1] 之后取这两个的最大值就是d

从状态管理到性能优化:全面解析 Android Compose

文章目录 引言一、Android Compose基本概念1.1 什么是Android Compose?1.2 Compose的优势1.3 如何在项目中使用Compose 二、Compose中的状态管理2.1 状态管理的重要性2.2 Compose中的状态和数据流2.3 使用State和MutableState处理状态2.4 通过ViewModel进行状态管理 三、Compose中的列表和滚动

Sentinel 高可用流量管理框架

Sentinel 是面向分布式服务架构的高可用流量防护组件,主要以流量为切入点,从限流、流量整形、熔断降级、系统负载保护、热点防护等多个维度来帮助开发者保障微服务的稳定性。 Sentinel 具有以下特性: 丰富的应用场景:Sentinel 承接了阿里巴巴近 10 年的双十一大促流量的核心场景,例如秒杀(即突发流量控制在系统容量可以承受的范围)、消息削峰填谷、集群流量控制、实时熔断下游不可用应

NGINX轻松管理10万长连接 --- 基于2GB内存的CentOS 6.5 x86-64

转自:http://blog.chinaunix.net/xmlrpc.php?r=blog/article&uid=190176&id=4234854 一 前言 当管理大量连接时,特别是只有少量活跃连接,NGINX有比较好的CPU和RAM利用率,如今是多终端保持在线的时代,更能让NGINX发挥这个优点。本文做一个简单测试,NGINX在一个普通PC虚拟机上维护100k的HTTP