八叉树建立地图并实现路径规划导航(上)

本文涉及的产品
资源编排,不限时长
简介: 八叉树建立地图并实现路径规划导航(上)

一、Octomap 安装并使用Octomap_Server


1.1 Apt 安装 Octomap 库


如果你不需要修改源码,可以直接安装编译好的 octomap 库,记得把 ROS 版本「kinetic」替换成你用的:


sudo apt-get install ros-kinetic-octomap*


上面这一行命令等价于安装以下的 octomap 组件:


sudo apt-get install ros-kinetic-octomap ros-kinetic-octomap-mapping ros-kinetic-octomap-msgs ros-kinetic-oc


注意:上面没有安装 ros-kinetic-octomap-server,原因是我要使用这个包来建图,并且需要修改它,所以在下一步我直接通过编译源码来安装它!


1.2 编译安装 OctomapServer 建图包


因为我要启用八叉树体素栅格的 RGB 颜色支持,需要修改源码,所以必须使用源码编译安装,过程如下:


创建编译用的工作空间
cd 你的一个目录/
# 创建工作空间
mkdir octomap_ws
cd octomap_ws/
# ROS 的工作空间必须包含 src 目录
mkdir src/
# 创建
catkin_make
# 记得 source 环境变量
source devel/setup.zsh


下载编译源码


cd src/
git clone https://github.com/OctoMap/octomap_mapping.git


返回你的工作空间主目录,安装下依赖,然后开始编译:


cd ../
rosdep install octomap_mapping
catkin_make


编译过程基本没有报错,如果你遇到问题,直接复制错误信息浏览器搜索解决,然后启动测试的 launch:


roslaunch octomap_server octomap_mapping.launch


没问题的话应该可以用 rostopic list 看到一个 octomap_full 的话题:


有这个话题说明这个建图包可以正常工作啦:)


1.3 Rviz 可视化 Octomap


ROS 中提供了一个 Rviz 可视化 Octomap 的插件,如果没有安装使用下面的命令即可:


sudo apt-get install ros-kinetic-octomap-rviz-plugins


安装后启动 Rviz,直接添加一个八叉树占用网格的类型,第一个是带颜色的类型,第二个不带颜色:


2021011921210451.png


建图节点启动后,选择话题名称为 octomap_full,即可显示出八叉树体素栅格,这是我的实验结果,我用的是第一个带颜色的类型:


2021011921212016.png


我把点云和体素栅格一起显示了,所以会重叠。这里要注意的是,如果你的点云显示不出来,要检查下「Global Options」的「Fixed Frame」有没有设置正确,我是设置的是 Robosense 雷达的 frame_id:「rslidar」。


1.4 启用 ColorOctomap


默认编译的 octomap 不能显示颜色,要开启颜色的支持,需要 2 个步骤,第一步编辑 OctomapServer.h 文件:


vim octomap_mapping/octomap_server/include/octomap_server/OctomapServer.h


打开下面 COLOR_OCTOMAP_SERVER 宏的注释即可:


/

// switch color here - easier maintenance, only maintain OctomapServer. 
// Two targets are defined in the cmake, octomap_server_color and octomap_server. One has this defined, and the other doesn't
// 打开这个注释
#define COLOR_OCTOMAP_SERVER


然后重新编译一遍源码:


cd octomap_ws/
catkin_make


第二步是在使用时,在 launch 文件中禁用 height_map,启用 colored_map,这个配置是我阅读源码查找的,因为官网文档很久没有更新了,一些参数配置方法需要通过阅读源码才能知道:


<param name = "height_map" value = "false" />
<param name = "colored_map" value = "true" />


比如以下是我实验用的 launch 文件:


<launch>
  <node pkg="octomap_server" type="octomap_server_node" name="octomap_server">
    <!-- resolution in meters per pixel -->
    <param name="resolution" value="0.10" />
    <!-- name of the fixed frame, needs to be "/map" for SLAM -->
    <!-- 
         增量式构建地图时,需要提供输入的点云帧和静态全局帧之间的 TF 变换
    -->
    <param name="frame_id" type="string" value="world" />
    <param name = "height_map" value = "false" />
    <param name = "colored_map" value = "true" /> 
    <!-- topic from where pointcloud2 messages are subscribed -->
    <!-- 要订阅的点云主题名称 /fusion_cloud -->
    <remap from="/cloud_in" to="/fusion_cloud" />
  </node>
</launch>


我设置了八叉树帧的 frame 为 rslidar,并将融合的点云话题 /fusion_cloud 作为节点的输入,我没有提供 TF,因为目前只是做了一个单帧的体素栅格构建,效果如下:


20210119212753873.png


二、增量构建八叉树地图步骤


为了能够让 octomap_server 建图包实现增量式的地图构建,需要以下 2 个步骤:


2021011921204714.png


2.1 配置 launch 启动参数


这 3 个参数是建图必备:


地图分辨率 resolution:用来初始化地图对象

全局坐标系 frame_id:构建的全局地图的坐标系

输入点云话题 cloud_in:作为建图的数据输入,建图包是把一帧一帧的点云叠加到全局坐标系实现建图


<launch>
  <node pkg="octomap_server" type="octomap_server_node" name="octomap_server">
    <!-- resolution in meters per pixel -->
    <param name="resolution" value="0.10" />
    <!-- 增量式构建地图时,需要提供输入的点云帧和静态全局帧之间的 TF 变换 -->
    <param name="frame_id" type="string" value="map" />
    <!-- 要订阅的点云主题名称 /fusion_cloud -->
    <remap from="/cloud_in" to="/fusion_cloud" />
  </node>
</launch>


以下是所有可以配置的参数:


  • frame_id (string, default: /map)


  • Static global frame in which the map will be published. A transform from sensor data to this frame needs to be available when dynamically building maps.


  • resolution (float, default: 0.05)


  • Resolution in meter for the map when starting with an empty map. Otherwise the loaded file’s resolution is used


  • height_map (bool, default: true)


  • Whether visualization should encode height with different colors


  • color/[r/g/b/a] (float)


  • Color for visualizing occupied cells when ~heigh_map=False, in range [0:1]


  • sensor_model/max_range (float, default: -1 (unlimited))


  • 动态构建地图时用于插入点云数据的最大范围(以米为单位),将范围限制在有用的范围内(例如5m)可以防止虚假的错误点。


  • sensor_model/[hit|miss] (float, default: 0.7 / 0.4)


  • 动态构建地图时传感器模型的命中率和未命中率


  • sensor_model/[min|max] (float, default: 0.12 / 0.97)


  • 动态构建地图时的最小和最大 clamp 概率


  • latch (bool, default: True for a static map, false if no initial map is given)


  • 不管主题是锁定发布还是每次更改仅发布一次,为了在构建地图(频繁更新)时获得最佳性能,请将其设置为 false,如果设置为 true,在每个地图上更改都会创建所有主题和可视化。


  • base_frame_id(string, default: base_footprint)


  • The robot’s base frame in which ground plane detection is performed (if enabled)


  • filter_ground(bool, default: false)


  • 动态构建地图时是否应从扫描数据中检测并忽略地平面,这会将清除地面所有内容,但不会将地面作为障碍物插入到地图中。如果启用此功能,则可以使用 ground_filter 对其进行进一步配置


  • ground_filter/distance (float, default: 0.04)


  • 将点(在 z 方向上)分割为接地平面的距离阈值,小于该阈值被认为是平面


  • ground_filter/angle (float, default: 0.15)


  • 被检测平面相对于水平面的角度阈值,将其检测为地面


  • ground_filter/plane_distance (float, default: 0.07)


  • 对于要检测为平面的平面,从 z = 0 到距离阈值(来自PCL的平面方程的第4个系数)


  • pointcloud_[min|max]_z (float, default: -/+ infinity)


  • 要在回调中插入的点的最小和最大高度,在运行任何插入或接地平面滤波之前,将丢弃此间隔之外的任何点。您可以以此为基础根据高度进行粗略过滤,但是如果启用了 ground_filter,则此间隔需要包括接地平面。


  • occupancy_[min|max]_z (float, default: -/+ infinity)


  • 最终 map 中要考虑的最小和最大占用单元格高度,发送可视化效果和碰撞 map 时,这会忽略区间之外的所有已占用体素,但不会影响实际的 octomap 表示。


  • filter_speckles(bool)


  • 是否滤除斑


2.2 要求 TF 变换


有了全局坐标系和每一帧的点云,但是建图包怎么知道把每一帧点云插入到哪个位置呢?


因为随着机器人的不断移动,会不断产生新的点云帧,每个点云帧在全局坐标系中插入的时候都有一个确定的位置,否则构建的地图就不对了,因此需要给建图包提供一个当前帧点云到全局坐标系的位姿,这样建图包才能根据这个位姿把当前获得的点云插入到正确的位置上。


在 ROS 中可以很方便的使用 TF 来表示这个变换,我们只需要在启动建图包的时候,在系统的 TF 树中提供「cloud frame -> world frame」的变换就可以了:


cloud frame -> world frame (static world frame, changeable with parameter frame_id)


注意:


cloud frame:就是输入点云话题中 head.frame_id,比如 Robosense 雷达的 frame_id = rslidar


world frame:这是全局坐标系的 frame_id,在启动 launch 中需要手动给定,比如我给的是 map


如果你给 world frame id 指定的是输入点云的 frame_id,比如 fusion_cloud.head.frame_id == wolrd_frame_id == rslidar,则只会显示当前帧的八叉树,而不会增量构建地图,这点要注意了,可以自己测试看看。


那么为了增量式建图,还需要在系统的 TF 树中提供「rslidar -> world」的变换,这个变换可以通过其他的 SLAM 等获得,比如我测试时候的一个 TF 树如下:


20210119205859439.png


我找了下源代码 OctomapServer.cpp 中寻找 TF 的部分:


/

tf::StampedTransform sensorToWorldTf;
  try {
    m_tfListener.lookupTransform(m_worldFrameId, cloud->header.frame_id, cloud->header.stamp, sensorToWorldTf);
  } catch(tf::TransformException& ex){
    ROS_ERROR_STREAM( "Transform error of sensor data: " << ex.what() << ", quitting callback");
    return;
  }


总体来说这个建图包使用起来还是挺简单的,最重要的就是要写清楚输入点云话题和 TF 变换。


小 Tips:没有 TF 怎么办?


我刚开始建图的时候,我同学录制的 TF 树中并没有「world -> rslidar」的变换,只有「world -> base_link」,所以为了能够测试增量式建图,因为我的点云帧的 frame_id 是 rslidar,因此我就手动发布了一个静态的「base_link -> rslidar」的变换:


rosrun tf2_ros static_transform_publisher 0 0 0 0 0 0 base_link rslidar


这样系统中就有「rslidar -> world」的变换了,但是我发的位姿都是 0,所以对建图测试没有影响,为了方便启动,放在 launch 中:


<node pkg = "tf2_ros" type = "static_transform_publisher" name = "dlonng_static_test_broadcaster" args = "0 0 0 0 0 0 base_link rslidar" />


如果你也遇到这个问题,可以试试发个静态 TF 做做测试,关于静态 TF 详细技术可以参考之前的文章:ROS 机器人技术 - 静态 TF 坐标帧


三、ColorOctomap 启用方法


为了显示 RGB 颜色,我分析了下源码,第一步修改头文件,打开注释切换地图类型:OctomapServer.h


// switch color here - easier maintenance, only maintain OctomapServer. 
// Two targets are defined in the cmake, octomap_server_color and octomap_server. One has this defined, and the other doesn't
// 打开这个注释
#define COLOR_OCTOMAP_SERVER
#ifdef COLOR_OCTOMAP_SERVER
  typedef pcl::PointXYZRGB PCLPoint;
  typedef pcl::PointCloud<pcl::PointXYZRGB> PCLPointCloud;
  typedef octomap::ColorOcTree OcTreeT;
#else
  typedef pcl::PointXYZ PCLPoint;
  typedef pcl::PointCloud<pcl::PointXYZ> PCLPointCloud;
  typedef octomap::OcTree OcTreeT;
#endif


CMakeList.txt 文件中有 COLOR_OCTOMAP_SERVER 的编译选项:


target_compile_definitions(${PROJECT_NAME}_color PUBLIC COLOR_OCTOMAP_SERVER)


OctomapServer.cpp 中有 colored_map 的参数:


m_useHeightMap = true;
m_useColoredMap = false;
m_nh_private.param("height_map", m_useHeightMap, m_useHeightMap);
m_nh_private.param("colored_map", m_useColoredMap, m_useColoredMap);


地图默认是按照高度设置颜色,如果要设置为带颜色的地图,就要禁用 HeightMap,并启用 ColoredMap:


if (m_useHeightMap && m_useColoredMap) {
    ROS_WARN_STREAM("You enabled both height map and RGB color registration. This is contradictory. Defaulting to height map.");
    m_useColoredMap = false;
  }


第二步、需要在 octomap_server 的 launch 文件中禁用 height_map,并启用 colored_map :


<param name="height_map" value="false" />
<param name="colored_map" value="true" />


2 个核心的八叉树生成函数 insertCloudCallback 和 insertScan 中有对颜色的操作:


#ifdef COLOR_OCTOMAP_SERVER
  unsigned char* colors = new unsigned char[3];
#endif
// NB: Only read and interpret color if it's an occupied node
#ifdef COLOR_OCTOMAP_SERVER 
        m_octree->averageNodeColor(it->x, it->y, it->z, /*r=*/it->r, /*g=*/it->g, /*b=*/it->b);
#endif


四、保存和显示地图


启动 octomap_server 节点后,可以使用它提供的地图保存服务,保存压缩的二进制存储格式地图:


octomap_saver mapfile.bt


保存一个完整的概率八叉树地图:


octomap_saver -f mapfile.ot


安装八叉树可视化程序 octovis 来查看地图:


sudo apt-get install octovis


安装后重启终端,使用以下命令显示一个八叉树地图:


octovis xxx.ot[bt]


相关实践学习
使用ROS创建VPC和VSwitch
本场景主要介绍如何利用阿里云资源编排服务,定义资源编排模板,实现自动化创建阿里云专有网络和交换机。
阿里云资源编排ROS使用教程
资源编排(Resource Orchestration)是一种简单易用的云计算资源管理和自动化运维服务。用户通过模板描述多个云计算资源的依赖关系、配置等,并自动完成所有资源的创建和配置,以达到自动化部署、运维等目的。编排模板同时也是一种标准化的资源和应用交付方式,并且可以随时编辑修改,使基础设施即代码(Infrastructure as Code)成为可能。 产品详情:https://www.aliyun.com/product/ros/
相关文章
|
算法 定位技术
八叉树建立地图并实现路径规划导航(下)
八叉树建立地图并实现路径规划导航(下)
2459 0
八叉树建立地图并实现路径规划导航(下)
|
3月前
|
数据采集 监控 安全
厂区地图导航制作:GIS技术与路径导航算法融合
在智能化、数字化时代,GIS技术为厂区的运营管理带来了革命性变化。本文探讨了如何利用GIS技术,通过数据采集、地图绘制、路径规划、位置定位和信息查询等功能,打造高效、精准的智能厂区地图导航系统,提升企业的竞争力和管理水平。
123 0
厂区地图导航制作:GIS技术与路径导航算法融合
|
5月前
|
定位技术
使用地图写路径规划
使用地图写路径规划
70 0
|
定位技术 内存技术
GIS空间分析 三维分析4 制作飞行动画
本文中,我们利用ArcScene软件用3种方法制作了飞行动画
362 0
|
8月前
|
定位技术
如何实现大地图的漫游功能
如何实现大地图的漫游功能
137 0
|
传感器 机器学习/深度学习 监控
【无人机控制】基于PRISM模型实现无人机目标搜索及避碰附matlab代码
【无人机控制】基于PRISM模型实现无人机目标搜索及避碰附matlab代码
|
机器学习/深度学习 传感器 编解码
【路径规划】基于A星算法结合floyd和动态窗口法实现机器人栅格地图路径规划附matlab代码
【路径规划】基于A星算法结合floyd和动态窗口法实现机器人栅格地图路径规划附matlab代码
|
定位技术
如何实现大地图漫游功能
最近有学员问如何实现割草游戏中的那种角色可以在一张大地图中四处跑动的功能,我研究了一下,发现其中有几个需要注意的地方,索性就写个教程记录一下。 想实现这个功能,最重要的是理解微信小游戏制作工具中的三个与相机有关的积木块的使用。 只需要借助这三个积木块,我们就能够轻松实现出一个大地图漫游的功能了。 下面,我们逐一来介绍一下这三块积木,以及其具体的应用。
172 0
|
机器学习/深度学习 传感器 算法
【机器人栅格地图】基于双向A星算法实现栅格地图机器人动态路径规划附matlab代码
【机器人栅格地图】基于双向A星算法实现栅格地图机器人动态路径规划附matlab代码
|
数据可视化 定位技术
GIS开发:客户端控制的地图样式
GIS开发:客户端控制的地图样式
113 0