gmapping 实现过程和数据走向

2024-09-06 04:12

本文主要是介绍gmapping 实现过程和数据走向,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

全局地图坐标系gmap_pose数据处理过程

初始化Map位姿

初始化地图坐标map:map是不会变化的,是gmapping 程序启动的位置姿态就是地图的原地

  // 地图原点设置为激光中心位置 启动程序时,设置原点GMapping::OrientedPoint gmap_pose(0, 0, 0);

激光初始化里程计的位姿

  // 获取初始姿态GMapping::OrientedPoint initialPose;if(!getOdomPose(initialPose, scan.header.stamp)){ROS_WARN("Unable to determine inital pose of laser! Starting point will be set to zero.");initialPose = GMapping::OrientedPoint(0.0, 0.0, 0.0);}

获取当前激光雷达时间下的odom->lase的tf来得到里程计的位姿
通过tf变换来计算里程计的位姿

/*** @brief 尝试获取指定时间下的机器人里程计位置* * @param gmap_pose 输出参数,用于存储计算出的里程计位置* @param t 指定的时间点* * @return 返回是否成功获取里程计位置*/
bool SlamGMapping::getOdomPose(GMapping::OrientedPoint& gmap_pose, const ros::Time& t)
{// 获取中心激光雷达在指定时间的位置centered_laser_pose_.stamp_ = t;// 使用TF变换获取激光雷达中心位置对应的里程计坐标系下的位置tf::Stamped<tf::Transform> odom_pose;try{tf_.transformPose(odom_frame_, centered_laser_pose_, odom_pose);}catch(tf::TransformException e){ROS_WARN("计算里程计位置失败,跳过扫描 (%s)", e.what());return false;}// 提取里程计坐标系下位置的姿态角double yaw = tf::getYaw(odom_pose.getRotation());// 构造并返回里程计位置gmap_pose = GMapping::OrientedPoint(odom_pose.getOrigin().x(),odom_pose.getOrigin().y(),yaw);return true;
}

传递位姿数据setPose—>getPose()

后面通过getPose来读取gmap_pose的位姿

  // 设置激光雷达扫描的姿态reading.setPose(gmap_pose);

gmap_pose数据保存和传递

// 防止多次定义
#ifndef ODOMETRYREADING_H
#define ODOMETRYREADING_H// 引入基本的传感器读数类和点类
#include <string.h>
#include <sensor/sensor_base/sensorreading.h>
#include <utils/point.h>
// 引入里程计传感器类
#include "odometrysensor.h"// 命名空间GMapping,用于地图构建相关功能
namespace GMapping{// 里程计读数类,继承自SensorReading
class OdometryReading: public SensorReading{public:// 构造函数,初始化里程计读数// 参数odo: 里程计传感器对象的指针// 参数time: 读数的时间戳,默认为0OdometryReading(const OdometrySensor* odo, double time=0);// 获取当前姿态inline const OrientedPoint& getPose() const {return m_pose;}// 获取当前速度inline const OrientedPoint& getSpeed() const {return m_speed;}// 获取当前加速度inline const OrientedPoint& getAcceleration() const {return m_acceleration;}// 设置当前姿态inline void setPose(const OrientedPoint& pose) {m_pose=pose;}// 设置当前速度inline void setSpeed(const OrientedPoint& speed) {m_speed=speed;}// 设置当前加速度inline void setAcceleration(const OrientedPoint& acceleration) {m_acceleration=acceleration;}protected:// 当前姿态OrientedPoint m_pose;// 当前速度OrientedPoint m_speed;// 当前加速度OrientedPoint m_acceleration;
};};
// 结束防止多次定义
#endif

激光雷达数据处理过程

laserCallback----->initMapper----->addScan---->updataMap

1. 数据预处理

  • 激光雷达数据获取:从激光雷达传感器获取原始的扫描数据。
  • 数据滤波:对原始数据进行滤波处理,去除噪声和异常点。

2. 粒子滤波初始化

  • 粒子初始化:在初始时刻,生成一组粒子(即机器人的可能位置),每个粒子代表机器人可能的一个状态。
  • 权重分配:为每个粒子分配初始权重,通常所有粒子的权重相等。

3. 运动模型更新

  • 预测步骤:根据机器人的运动模型(如里程计数据),预测每个粒子在下一个时刻的位置。
  • 粒子扩散:由于运动模型存在误差,粒子会在预测位置周围进行一定程度的扩散,以模拟不确定性。

4. 观测模型更新

  • 激光数据匹配:将当前时刻的激光雷达数据与地图(由粒子表示)进行匹配,计算每个粒子的似然度。
  • 权重更新:根据激光数据的匹配结果,更新每个粒子的权重。匹配效果好的粒子权重增加,匹配效果差的粒子权重减少。

5. 重采样

  • 粒子重采样:根据更新后的粒子权重,进行重采样。权重高的粒子被保留或复制,权重低的粒子被淘汰,以确保粒子集能够更好地代表机器人的真实状态。

6. 地图更新

  • 地图构建:根据重采样后的粒子集,更新地图。每个粒子对应一个局部地图,通过融合所有粒子的局部地图,得到全局地图。
  • 地图优化:对地图进行优化,去除噪声点,平滑地图边界。

7. 循环迭代

  • 时间更新:重复上述步骤,随着时间的推移,不断更新粒子集和地图,直到完成整个环境的建图。

8. 结果输出

  • 定位结果:输出机器人的最终位置估计。
  • 地图输出:输出最终构建的地图。

这篇关于gmapping 实现过程和数据走向的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C++使用栈实现括号匹配的代码详解

《C++使用栈实现括号匹配的代码详解》在编程中,括号匹配是一个常见问题,尤其是在处理数学表达式、编译器解析等任务时,栈是一种非常适合处理此类问题的数据结构,能够精确地管理括号的匹配问题,本文将通过C+... 目录引言问题描述代码讲解代码解析栈的状态表示测试总结引言在编程中,括号匹配是一个常见问题,尤其是在

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

Java覆盖第三方jar包中的某一个类的实现方法

《Java覆盖第三方jar包中的某一个类的实现方法》在我们日常的开发中,经常需要使用第三方的jar包,有时候我们会发现第三方的jar包中的某一个类有问题,或者我们需要定制化修改其中的逻辑,那么应该如何... 目录一、需求描述二、示例描述三、操作步骤四、验证结果五、实现原理一、需求描述需求描述如下:需要在

JavaScript中的reduce方法执行过程、使用场景及进阶用法

《JavaScript中的reduce方法执行过程、使用场景及进阶用法》:本文主要介绍JavaScript中的reduce方法执行过程、使用场景及进阶用法的相关资料,reduce是JavaScri... 目录1. 什么是reduce2. reduce语法2.1 语法2.2 参数说明3. reduce执行过程

如何使用Java实现请求deepseek

《如何使用Java实现请求deepseek》这篇文章主要为大家详细介绍了如何使用Java实现请求deepseek功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1.deepseek的api创建2.Java实现请求deepseek2.1 pom文件2.2 json转化文件2.2

python使用fastapi实现多语言国际化的操作指南

《python使用fastapi实现多语言国际化的操作指南》本文介绍了使用Python和FastAPI实现多语言国际化的操作指南,包括多语言架构技术栈、翻译管理、前端本地化、语言切换机制以及常见陷阱和... 目录多语言国际化实现指南项目多语言架构技术栈目录结构翻译工作流1. 翻译数据存储2. 翻译生成脚本

如何通过Python实现一个消息队列

《如何通过Python实现一个消息队列》这篇文章主要为大家详细介绍了如何通过Python实现一个简单的消息队列,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录如何通过 python 实现消息队列如何把 http 请求放在队列中执行1. 使用 queue.Queue 和 reque

Python如何实现PDF隐私信息检测

《Python如何实现PDF隐私信息检测》随着越来越多的个人信息以电子形式存储和传输,确保这些信息的安全至关重要,本文将介绍如何使用Python检测PDF文件中的隐私信息,需要的可以参考下... 目录项目背景技术栈代码解析功能说明运行结php果在当今,数据隐私保护变得尤为重要。随着越来越多的个人信息以电子形

使用 sql-research-assistant进行 SQL 数据库研究的实战指南(代码实现演示)

《使用sql-research-assistant进行SQL数据库研究的实战指南(代码实现演示)》本文介绍了sql-research-assistant工具,该工具基于LangChain框架,集... 目录技术背景介绍核心原理解析代码实现演示安装和配置项目集成LangSmith 配置(可选)启动服务应用场景