RTAB-Map 闭环检测

2024-05-31 22:32
文章标签 检测 map 闭环 rtab

本文主要是介绍RTAB-Map 闭环检测,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

欢迎访问 https://cgabc.xyz/posts/999fccc5/,持续更新

1. 概述

主要特点:

  • 基于外观(Appearance-Based),通过图像相似度查找回环
  • 贝叶斯滤波算法,估计回环的概率
  • 增量式在线构建视觉词典或词袋,针对一个特定环境不需要预训练过程
  • 内存管理模型,保证实时在线运行

代码主要过程:

  • RTABMap(闭环检测)主入口函数 Rtabmap::process
  • 输入图像image及其idheader.seq)被封装到SensorData
  • 内存更新(Memory::update)
    • 创建签名(Memory::createSignature)
    • Add Signature To STM(Memory::addSignatureToStm)
    • Weight Update, Rehearsal(Memory::rehearsal)
    • Transfer the oldest signature from STM to WM(Memory::moveSignatureToWMFromSTM)
  • 贝叶斯滤波器更新
    • 计算似然(Memory::computeLikelihood)
    • 调整似然(Rtabmap::adjustLikelihood)
    • 计算后验(BayesFilter::computePosterior)
    • 选择最高的闭环假设
  • RETRIEVAL
    • Loop closure neighbors reactivation
    • Load signatures from the database, from LTM to WM(Memory::reactivateSignatures)
  • Update loop closure links: make neighbors of the loop closure in RAM
  • TRANSFER: move the oldest signature from STM to LTM

算法主要流程:

内存管理模型:


回环检测(若不考虑内存管理)过程:

2. 内存更新

2.1 创建签名

代码在 Memory::createSignature 中,其主要过程为

  • 词典更新(VWDictionary::update) 线程
    • 构建FLANN索引
      • 根据描述子构建KDTree索引词典 (_flannIndex->buildKDTreeIndex, kNNFlannKdTree)
      • KDTree的创建基于分层k-mean聚类
    • 更新_dataTree
  • 角点(GFTT)检测(Feature2D::generateKeypoints)
    • 均匀分布(gridRows_, gridCols_)
    • 限制点数(maxFeatures_)
    • 亚像素提取(cv::cornerSubPix)
  • 描述子(BRIEF)计算(Feature2D::generateDescriptors)
  • quantize descriptors to vocabulary(VWDictionary::addNewWords)
    • 描述子匹配(descriptors – dataTree),并计算距离

    • 添加单词 或 参考+1

      • badDist=true(匹配数量特别少,或 NNDR(neareast neighbor distance ratio) 大于 T NNDR T_{\text{NNDR}} TNNDR 阈值_nndrRatio),然后创建VisualWord并添加到_visualWords

      NNDR = dist of the neareast neighbor dist of the second-neareast neighbor \text{NNDR} = \frac{\text{dist of the neareast neighbor}}{\text{dist of the second-neareast neighbor}} NNDR=dist of the second-neareast neighbordist of the neareast neighbor

      • badDist=false,VWDictionary::addWordRef_references+1
  • 创建签名(new Signature)

2.2 添加签名到STM

代码主要在 Memory::addSignatureToStm 中。

  • 更新neighbors,添加链接(signature->addLink)
  • 添加签名ID到_stMem

2.3 更新权重

代码主要在 Memory::rehearsal 中。

  • signature与STM中最新的签名比较,计算相似度RehearsalSimilarity(float sim = signature->compareTo(*sB))

similarity = pairs totalWords \text{similarity} = \frac{\text{pairs}}{\text{totalWords}} similarity=totalWordspairs

  • 相似度 > 阈值_similarityThreshold,假设合并(Memory::rehearsalMerge),更新权重

    • 更新权重(signature->setWeight)

    w A = w A + w B + 1 w_A = w_A + w_B + 1 wA=wA+wB+1

    • 新旧签名合并(Memory::rehearsalMerge)
      • 新签名添加Link
      • 删除旧签名(moveToTrash)

2.4 签名转移 (STM->WM)

Transfer the oldest signature of the short-term memory to the working memory

  • _maxStMemSize = 10
  • Memory::moveSignatureToWMFromSTM:_workingMem.insert_stMem.erase

3. 贝叶斯滤波器更新

  • 计算似然(Memory::computeLikelihood),得到 rawLikelihood

    • 算法一:tf-idf (term frequency–inverse document frequency)

    tf-idf = n w i + log ⁡ N n w n i \text{tf-idf} = \frac{n_{wi} + \log \frac{N}{n_w}}{n_i} tf-idf=ninwi+lognwN

    • 算法二:相似度 RehearsalSimilarity
  • 调整似然(Rtabmap::adjustLikelihood) s j s_j sj,得到 likelihood

    • 依据 似然均值 μ \mu μ 和 似然标准差 σ \sigma σ

    likelihood = { s j − σ μ , if  s j ≥ μ + σ 1 , otherwise . \text{likelihood} = \begin{cases} \frac{s_j - \sigma}{\mu}, \quad \text{if} \ s_j \geq \mu + \sigma \\ 1, \quad\quad\quad \text{otherwise}. \end{cases} likelihood={μsjσ,if sjμ+σ1,otherwise.

  • 计算后验(BayesFilter::computePosterior),得到 posterior

    • 预测(Prediction : Prior*lastPosterior)得到_prediction
    • 更新后验(BayesFilter::updatePosterior)
    • 计算先验(prior = _prediction * posterior)
    • 计算后验
    • 后验归一化

posterior = likelihood × prior \text{posterior} = \text{likelihood} \times \text{prior} posterior=likelihood×prior

4. 回环假设选择

Select the highest hypothesis

  • 根据后验posterior选择最高的假设_highestHypothesis
  • 接受回环,条件如下:
    • _highestHypothesis > _loopThr
    • 对极几何 检查(_epipolarGeometry->check)
      • 单词匹配对数量(EpipolarGeometry::findPairsUnique) > _matchCountMinAccepted
      • 对极约束内点数(EpipolarGeometry::findFFromWords) > _matchCountMinAccepted(RANSAC方法计算基础矩阵)

5. 取回 Retrieval (LTM->WM)

对于形成回环概率最高的定位点,将他那些没有在WM中的邻接定位点,从LTM中取回放入到WM中。

  • Loop closure neighbors reactivation
    • time
    • space
  • Update planned path and get next nodes to retrieve
  • Load signatures from the database (LTM->WM)

6. 转移 Transfer (STM->LTM)

If time allowed for the detection exceeds the limit of real-time, move the oldest signature with less frequency entry (from X oldest) from the short term memory to the long term memory.

具有最低权重的定位点中,存储时间最长的将被转移到LTM(数据库SQLite)中。

参考文献

[1] Appearance-Based Loop Closure Detection for Online Large-Scale and Long-Term Operation
[2] Fast and Incremental Method for Loop-Closure Detection Using Bags of Visual Words

这篇关于RTAB-Map 闭环检测的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Linux系统性能检测命令详解

《Linux系统性能检测命令详解》本文介绍了Linux系统常用的监控命令(如top、vmstat、iostat、htop等)及其参数功能,涵盖进程状态、内存使用、磁盘I/O、系统负载等多维度资源监控,... 目录toppsuptimevmstatIOStatiotopslabtophtopdstatnmon

C++ 检测文件大小和文件传输的方法示例详解

《C++检测文件大小和文件传输的方法示例详解》文章介绍了在C/C++中获取文件大小的三种方法,推荐使用stat()函数,并详细说明了如何设计一次性发送压缩包的结构体及传输流程,包含CRC校验和自动解... 目录检测文件的大小✅ 方法一:使用 stat() 函数(推荐)✅ 用法示例:✅ 方法二:使用 fsee

OpenCV实现实时颜色检测的示例

《OpenCV实现实时颜色检测的示例》本文主要介绍了OpenCV实现实时颜色检测的示例,通过HSV色彩空间转换和色调范围判断实现红黄绿蓝颜色检测,包含视频捕捉、区域标记、颜色分析等功能,具有一定的参考... 目录一、引言二、系统概述三、代码解析1. 导入库2. 颜色识别函数3. 主程序循环四、HSV色彩空间

Java中Map.Entry()含义及方法使用代码

《Java中Map.Entry()含义及方法使用代码》:本文主要介绍Java中Map.Entry()含义及方法使用的相关资料,Map.Entry是Java中Map的静态内部接口,用于表示键值对,其... 目录前言 Map.Entry作用核心方法常见使用场景1. 遍历 Map 的所有键值对2. 直接修改 Ma

Java中JSON格式反序列化为Map且保证存取顺序一致的问题

《Java中JSON格式反序列化为Map且保证存取顺序一致的问题》:本文主要介绍Java中JSON格式反序列化为Map且保证存取顺序一致的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未... 目录背景问题解决方法总结背景做项目涉及两个微服务之间传数据时,需要提供方将Map类型的数据序列化为co

Java使用Stream流的Lambda语法进行List转Map的操作方式

《Java使用Stream流的Lambda语法进行List转Map的操作方式》:本文主要介绍Java使用Stream流的Lambda语法进行List转Map的操作方式,具有很好的参考价值,希望对大... 目录背景Stream流的Lambda语法应用实例1、定义要操作的UserDto2、ListChina编程转成M

使用Python实现IP地址和端口状态检测与监控

《使用Python实现IP地址和端口状态检测与监控》在网络运维和服务器管理中,IP地址和端口的可用性监控是保障业务连续性的基础需求,本文将带你用Python从零打造一个高可用IP监控系统,感兴趣的小伙... 目录概述:为什么需要IP监控系统使用步骤说明1. 环境准备2. 系统部署3. 核心功能配置系统效果展

SpringBoot如何通过Map实现策略模式

《SpringBoot如何通过Map实现策略模式》策略模式是一种行为设计模式,它允许在运行时选择算法的行为,在Spring框架中,我们可以利用@Resource注解和Map集合来优雅地实现策略模式,这... 目录前言底层机制解析Spring的集合类型自动装配@Resource注解的行为实现原理使用直接使用M

C++ 各种map特点对比分析

《C++各种map特点对比分析》文章比较了C++中不同类型的map(如std::map,std::unordered_map,std::multimap,std::unordered_multima... 目录特点比较C++ 示例代码 ​​​​​​代码解释特点比较1. std::map底层实现:基于红黑

JavaScript中的Map用法完全指南

《JavaScript中的Map用法完全指南》:本文主要介绍JavaScript中Map用法的相关资料,通过实例讲解了Map的创建、常用方法和迭代方式,还探讨了Map与对象的区别,并通过一个例子展... 目录引言1. 创建 Map2. Map 和对象的对比3. Map 的常用方法3.1 set(key, v