ORB-SLAM2 论文&代码学习 —— LoopClosing 线程
转载请注明出处,谢谢
原创作者:Mingrui
原创链接:https://www.cnblogs.com/MingruiYu/p/12369339.html
本文要点:
- ORB-SLAM2 LoopClosing 线程 论文内容介绍
- ORB-SLAM2 LoopClosing 线程 代码结构介绍
写在前面
之前的 ORB-SLAM2 系列文章中,我们已经对 Tracking 线程和 LocalMapping 进行了介绍。我们将在本文中,对 ORB-SLAM2 系统的 LoopClosing 线程进行介绍。
依旧祭出该图,方便查看:
也再次献上我绘制的程序导图全图:ORB-SLAM2 程序导图
老规矩,还是分两部分:以 ORB-SLAM 论文为参考 和 以 ORB-SLAM2 代码(程序导图)为参考。
以 ORB-SLAM 论文为参考
LoopClosing 线程的大致步骤如下:
- 接收 LoopClosing 送来的筛选处理后的 KF
- 检测出一批 Candidate KFs
- 计算 Sim3,确定最终的 Loop KF
- 进行回环融合
- 优化 Essential Graph
这些步骤可以归为两类:
- 回环检测 Loop Dectection
- 回环校正 Loop Correction
下面我们对每一个步骤进行详细的介绍。
接收 LoopClosing 送来的筛选处理后的 KF
在之前的 LocalMapping 线程中,会对 KFs 进行筛选,最终在 Map 中保留必要的 KFs。经过筛选后的 KFs,会送入 LoopClosing 线程,每一个新送入的 Current KF,都会检测在 KF Database 中,有没有以前路过的 KF,与 Current KF 相匹配,从而可以进行回环校正。
检测出一批 Candidate KFs
这一步的目的是初步筛选出一批 Candidate KFs。
首先,会依据 KF 的 BoW 计算 Current KF 与其每一个 Covisible KF 之间的相似分数,取最低分作为 \(s_{min}\),Candidate KFs 与 Current KF 之间的相似分数至少要大于 \(s_{min}\)(动态计算选择 Candidate KFs 的阈值)。
另外,Current KF 的 Covisible KF 是不能成为 Candidate KFs 的。
再之后,还要进行连续性检测,进一步筛选 Candidate KFs:该 Candidate KF 需要在 Covisibility Graph 中与三个以上 Candidate KFs 以 Group 形式相连。
(以上部分论文中说的比较含糊,看代码会相对清晰,下文会详述)
计算 Sim3,确定最终的 Loop KF
在回环校正的时候,我们需要对 Current KF 和 Loop KF 之间的相对误差进行描述。但是注意,在单目 SLAM 中,是有7个自由度的:6 + 尺度。也就是说,对于单目 SLAM,在运行过程中的累计漂移,除了平移漂移,旋转漂移之外,还会有尺度漂移。可能跟着跟着轨迹和 Map 的尺度都整体缩小了。如果仅计算 SE3 的话,没法很好的对回环进行校正,所以提出了 Sim3(相似变换群):
\(\operatorname{sim}(3)=\left\{\left[\begin{array}{cc} {S=} & {s \boldsymbol{R}} & {t} \\ {\mathbf{0}^{T}} & {1} \end{array}\right] \in \mathbb{R}^{4 \times 4}\right\}\)
其中 \(S\) 就是相似变换,其在欧式变换的基础上加上了一个尺度因子 \(s\)。
ORB-SLAM 会计算 Current KF 与 Candidiate KF 之间的相似变换,并据此确定最终的 Loop OK,其步骤如下:
- 对 Current KF 与 Candidate KF 的 FeaturePoints 进行 BoW 匹配(要求是与 MapPoints 链接的 FeaturePoints),从而的到 MapPoints 与 MapPoints 之间的匹配(3D - 3D)。
- RANSAC 迭代计算每一个 Candidate KF 与 Current KF 之间的相似变换。
- 当找到一个有足够多 inliers 的相似变换后,进行优化;根据优化得到的相似变换,可以找到更多的 Current KF 与 该 Candidate KF 的 3D - 3D 匹配。
- 再进行优化,如果又有足够多 inliers,则认为该相似变换满足要求,也就找到一个满足要求的 Loop KF。
至此,就为 Current KF 确定了一个 Loop KF,Loop Detection 部分就完成了,之后就进入 Loop Correction。
回环融合
回环融合分以下几步:
- 校正 Current KF 及其 Convisible KFs 的 SE3 位姿
- 根据 Loop KF 的位姿 和 Loop KF 与 Current KF 之间的相似变换,对 Current KF 的位姿进行校正。
- 同时,也对 Current KF 的 Convisible KFs 的位姿进行校正(在 Current KF 的校正位姿基础上,叠加 Current KF 与 Convisible KFs 之间的相对位姿)。
- 将 Loop KF 及其 Convisible KFs 与 Current KF 及其 Convisible KFs 的进行 MapPoints 融合
- 将 Loop KF 及其 Convisible KFs 所含的 MapPoints 与 Current KF 及其 Convisible KFs 的 FeaturePoints 进行匹配(通过投影找匹配),匹配上了就进行 MapPoints 融合(此处的融合和 LocalMapping 中的 MapPoinits 融合采取同样方式)。
- 更新 Convisibility Graph
- 相关 KFs 更新其 Convisible KFs
- 添加回环边
优化 Essential Graph
上述回环校正只对 Current KF 附近的 KFs 进行了校正,但是累计误差是慢慢积累的,一路上每个 KF 都应该接受校正。
ORB-SLAM 采用了对 Essential Graph 进行位姿图优化的方式。该优化是 Sim3 优化,以便校正尺度漂移。
对 KFs 校正后,还有对 MapPoints 的坐标进行校正。ORB-SLAM 采用的方式比较粗暴:观测到该 MapPoints 的某 KF 的校正后位姿 * 该 MapPoints 与该 KF 校正前的相对坐标,就是 MapPoints 校正后的坐标。
以 ORB-SLAM2 代码(程序导图)为参考
请点击 ORB-SLAM2 程序导图链接(文首)查看清晰全图
与 LocalMapping 线程一样,LoopClosing 线程干的第一件事,也是检查新 KF 队列中有没有未处理的 KF,有的话,取出队首元素作为 Current KF
回环检测 —— 检测 Candidate KFs
LoopClosing::DetectLoop(),首先,之前刚进行完回环检测的10帧 KFs 内,不进行回环检测,且这10帧 KFs 不会添加进 KF Database。
BoW 检索
首先,计算 \(s_{min}\)。
之后,KeyFrameDatabase::DetectLoopCandidates() 筛选,除了论文中提到的根据 \(s_{min}\) 筛选,还要求 Candidate KF 与 Current KF 的共同 word 数 > 0.8 * 任意 KF 与 Current KF 的最多共同 word 数(但不包括 Current KF 的 Covisible KFs)。
之后,还要再筛选一次:队上述所有符合要求的 KFs,还有将它和其所有 Covisible KFs 组成一组,计算该组的相似分数之和,并且找出该组内相似分数最高的 KF。将所有 (总分数 > 0.75 * 最高组分数) 的组的最高分 KF 选入 Candidate KFs,送入下一步(这一步的意义是,如果单蹦出一 KF 相似分数很高,但它附近的 KFs 的相似分数都很低,那么可能这 KF 分高是不可靠的,需要筛掉)。
连续性检测
进一步筛选 Candidate KFs:进行连续性检测,该 Candidate KF 需要在 Covisibility Graph 中与三个以上 Candidate KFs 以 Group 形式相连。
什么是以 Group 形式相连呢?假设 KF1 及其 Covisible KFs 组成 Group1,KF2 及其 Covisible KFs 组成 Group2;如果 Group1 与 Group2 含有相同的 KF,则两个 Group 相连。
所以如果一个 Candidate KF 的 Group 与 另外 3 个 Candidate KFs 的 Groups 相连,则通过筛选,进入下一环节。
回环检测 —— 计算 Sim3,确定最终的 Loop KF
LoopClosing::ComputeSim3(),上一步中,会得到好几个 Candidate KF,这一步的目的之一是的到最终的 Loop KF,另外,还要计算的到 Current KF 与 Loop KF 之间的相似变换。
注: 从上述回环检测的步骤可以看出,ORB-SLAM2 在检测回环时,分层地进行了大量的筛选步骤。这样,一是为了保证 Loop KF 的准确性,二是避免出现大量相似回环:一般来说,因为场景是连续的,所以出现回环的地方,一般会出现一连串差不多的回环,这些筛选的步骤避免了添加重复的回环,也是减轻 LoopClosing 线程的负担。
回环校正
LoopClosing::CorrectLoop(),首先让通知 LocalMapping 暂停,防止插入新的 KF,并等待 LocalMapping 线程停止。期间还要停止正在进行的 Global BA(如果有的话)。
回环融合
与论文中描述的一致。
注意其中 MapPoints 的融合:将 Loop KF 所含的 MapPoints 与 Current KF 的 FeaturePoints 进行匹配,匹配上的话就将该 MapPoint 与 那个 FeaturePoint 链接上,但如果 Current KF 的该 FeaturePoint 已经链接上了某个 MapPoint,则用 Loop KF 的 MapPoint 替换掉原来链接的 MapPoint。
之后,再将 Loop KF 的 Covisible KFs 与 Current KF 的 Covisible KFs 两部分的 MapPoints 做匹配,使用 ORBmatcher::Fuse()。
最后,更新这局部的 Convisibility Graph。
优化 Essential Graph
Optimizer::OptimizeEssentialGraph()
注意,这之后个 Loop KF -> AddLoopEdge(Current KF) 和 Current KF -> AddLoopEdge(Loop KF) 只是记录回环边。Convisibility Graph 在之前已经更新过了。
Global BA
上述步骤全部进行完了之后,精度已经很高了,但 ORB-SLAM2 还是选择在最后再进行一次 Global BA 锦上添花(ORB-SLAM1 中似乎没有这步)。注意,为了不影响主要3个线程的工作,这里创建了第4个线程,专门进行 Global BA。但该 Global BA 随时可能被打断,只有在系统特别闲的时候才会运行。
ORB-SLAM2 系列博文
[ORB-SLAM2 系列博文](https://www.cnblogs.com/MingruiYu/tag/ORB-SLAM2/)
ORB-SLAM2 论文&代码学习 —— LoopClosing 线程的更多相关文章
- ORB-SLAM2 论文&代码学习 —— LocalMapping 线程
转载请注明出处,谢谢 原创作者:Mingrui 原创链接:https://www.cnblogs.com/MingruiYu/p/12360913.html 本文要点: ORB-SLAM2 Local ...
- ORB-SLAM2 论文&代码学习 ——Tracking 线程
本文要点: ORB-SLAM2 Tracking 线程 论文内容介绍 ORB-SLAM2 Tracking 线程 代码结构介绍 写在前面 上一篇文章中我们已经对 ORB-SLAM2 系统有了一个概览性 ...
- ORB-SLAM2 论文&代码学习 —— 单目初始化
转载请注明出处,谢谢 原创作者:Mingrui 原创链接:https://www.cnblogs.com/MingruiYu/p/12358458.html 本文要点: ORB-SLAM2 单目初始化 ...
- ORB-SLAM2 论文&代码学习 —— 概览
转载请注明出处,谢谢 原创作者:MingruiYU 原创链接:https://www.cnblogs.com/MingruiYu/p/12347171.html *** 本文要点: ORB-SLAM2 ...
- ORB SLAM2在Ubuntu 16.04上的运行配置
http://www.mamicode.com/info-detail-1773781.html 安装依赖 安装OpenGL 1. 安装opengl Library$sudo apt-get inst ...
- python自动化开发学习 进程, 线程, 协程
python自动化开发学习 进程, 线程, 协程 前言 在过去单核CPU也可以执行多任务,操作系统轮流让各个任务交替执行,任务1执行0.01秒,切换任务2,任务2执行0.01秒,在切换到任务3,这 ...
- 手写一个线程池,带你学习ThreadPoolExecutor线程池实现原理
摘要:从手写线程池开始,逐步的分析这些代码在Java的线程池中是如何实现的. 本文分享自华为云社区<手写线程池,对照学习ThreadPoolExecutor线程池实现原理!>,作者:小傅哥 ...
- luogg_java学习_12_线程
本文为博主辛苦总结,希望自己以后返回来看的时候理解更深刻,也希望可以起到帮助初学者的作用. 转载请注明 出自 : luogg的博客园 谢谢配合! 线程 程序.进程.线程的概念 程序:我们用程序设计语言 ...
- Java并发包源码学习之线程池(一)ThreadPoolExecutor源码分析
Java中使用线程池技术一般都是使用Executors这个工厂类,它提供了非常简单方法来创建各种类型的线程池: public static ExecutorService newFixedThread ...
随机推荐
- 【JDK1.8】 Java小白的源码学习系列:HashMap
目录 Java小白的源码学习系列:HashMap 官方文档解读 基本数据结构 基本源码解读 基本成员变量 构造器 巧妙的tableSizeFor put方法 巧妙的hash方法 JDK1.8的putV ...
- 团队项目-Beta冲刺1(七个小矮人)
团队项目-Beta冲刺1(七个小矮人) 一.格式描述 这个作业属于哪个课程 https://edu.cnblogs.com/campus/xnsy/GeographicInformationScien ...
- C#的WinForm窗体美化
为了帮助用户追求美观,.NET 4.0 专门为对此有需求的人提供了IrisSkin4.dll皮肤引用集,里面封装了许多对窗体重新描绘的方法,再搭配上WinForm特有的 .ssk 文件,就可以实现窗体 ...
- 初入计科,首次接触C的感受
1 你对计算机科学与技术专业了解是怎样? 答:一开始我对这个专业并无了解,觉得无非是把电脑给学透.经过一周的学习后,我深刻地感觉自己对这个专业深深的误解. 通过翻阅书籍,上网浏览相关信息,我认为该专业 ...
- Web自动化测试项目(三)用例的组织与运行
一.Unittest用例组织 在test_case目录下创建test*.py,组织测试用例 ├── test_case │ ├── __init__.py │ └── test_login.p ...
- Ceph 存储集群-低级运维
低级集群运维包括启动.停止.重启集群内的某个具体守护进程:更改某守护进程或子系统配置:增加或拆除守护进程.低级运维还经常遇到扩展.缩减 Ceph 集群,以及更换老旧.或损坏的硬件. 一.增加/删除 O ...
- python学习记录(四)
0828--https://www.cnblogs.com/fnng/archive/2013/04/18/3029807.html 0828--https://www.cnblogs.com/fnn ...
- CCF_201312-4_有趣的数
dp题,dp[i][j]代表i位数,j状态的数量.其中,j 的状态表示值有6种. 0 1 2 √ j = 0 3 01 02 √ j = 1 03 12 13 23 √ j = 2 0 ...
- 2020你还不会Java8新特性?
Java8(1)新特性介绍及Lambda表达式 前言: 跟大娃一块看,把原来的电脑拿出来放中间看视频用 --- 以后会有的课程 难度 深入Java 8 难度1 并发与netty 难度3 JVM 难度4 ...
- tomcat-windows10环境搭建
1.进入Tomcat官网Apache Tomcat® - Welcome! 2.根据操作系统选择合适的版本下载 zip用于windows操作系统, tar.gz用于unix和linux操作系统 Bin ...