开源自主导航小车MickX4(六)cartographer 室外2D建图

2023-05-16

开源自主导航小车MickX4(六)cartographer 室外2D建图

  • 1 cartographer环境安装
    • 1.1 安装ceres库
    • 1.2 安装cartographer
  • 2 cartographer demo测试
    • 2.1 启动2D 建图demo
    • 2.2 启动2D 定位demo
    • 2.3 小结
  • 3 在小车上实现2D室外建图
  • 4 在小车上基于已有2D地图定位
  • 5 大范围场景2D建图
  • 参考资料

1 cartographer环境安装

cartographer安装主要分为三个部分 ceres优化库[5]、cartographer[4]和 cartographer_ros[3]

1.1 安装ceres库

这里由于我需要使用其他的视觉SLAM框架,因此我安装的是1.14.0的版本,如果使用的是ubuntu18,clone 最新版的ceres库就行

下载编译ceres库

sudo apt-get install cmake libeigen3-dev libatlas-base-dev liblapack-dev libgflags-dev libgoogle-glog-dev libgtest-dev libsuitesparse-dev
cd ~/soft
wget http://ceres-solver.org/ceres-solver-1.14.0.tar.gz
tar zxf ceres-solver-1.14.0.tar.gz
cd ceres-solver-1.14.0
mkdir build
cd build
cmake ..
make -j3
sudo make install

ubuntu18安装最新库的使用下面命令

git clone https://ceres-solver.googlesource.com/ceres-solver
cd ceres-solver
mkdir build
cd build
cmake ..
make -j3
sudo make install

1.2 安装cartographer

安装依赖项目

sudo apt-get update
sudo apt-get install -y python-wstool python-rosdep ninja-build stow
sudo apt-get install -y python-wstool python-rosdep ninja-build
sudo apt-get install -y cmake g++ git google-mock libboost-all-dev libcairo2-dev libeigen3-dev libgflags-dev libgoogle-glog-dev liblua5.2-dev libprotobuf-dev libsuitesparse-dev libwebp-dev ninja-build protobuf-compiler python-sphinx

下载源代码

cd catkin_cartographer_ws/src

git clone https://github.com/cartographer-project/cartographer.git
git clone https://github.com/googlecartographer/cartographer_ros.git
  
cd ~/catkin_cartographer_ws/src/cartographer/scripts
./install_proto3.sh
./install_abseil.sh 

catkin_make_isolated --install --use-ninja 

如果提示缺少其他的依赖项可以使用 cartographer/scripts 目录下的脚本进行安装。这里 cartographer 使用的编译命令不是 catkin_make 因此建议不与其他工作空间放一起。
在这里插入图片描述
环境安装尽量多参考官网的教程,官方的教程在更新。也可参考hitcm博主[1]的教程或者我以前的博客[2] ( cartographer 在机器人上运行建图与定位)。

2 cartographer demo测试

通过官方提供的bag测试 cartographer 安装是否正确,下载测试包:

  • 百度云(提取码:46fq )
  • 官网链接

2.1 启动2D 建图demo

source devel_isolated/setup.bash
roslaunch cartographer_ros demo_backpack_2d.launch bag_filename:=/home/administrator/cartograph/b2-2016-04-05-14-44-52.bag

bag_filename表示的是ROS bag的数据包。

注意:由于发现 offline_backpack_2d.launch 这个文件没有启动地图保存服务。这里把官网上的 offline_backpack_2d.launch 文件替换为了 demo_backpack_2d.launch。

cartograph_数据集2D建图(x8)

先将地图保存为 .pbstream 文件

rosservice call /write_state ~/cartograph_test.pbstream

用 cartographer 自带的转换节点将.pbstream 文件转化为pgm和yaml文件

source devel_isolated/setup.bash
rosrun cartographer_ros cartographer_pbstream_to_ros_map -pbstream_filename /home/administrator/cartograph_test.pbstream -map_filestem /home/administrator/cartograph_test

启动节点以后可以看到文件夹下生成的pgm 和 yaml文件
在这里插入图片描述
但是如果你需要使用 cartographer 进行定位的话,就没有必要去转换为pgm格式的。

2.2 启动2D 定位demo

接下来我们使用已有的地图进行定位

source devel_isolated/setup.bash
roslaunch cartographer_ros demo_backpack_2d_localization.launch load_state_filename:=/home/crp/ cartograph_test.pbstream bag_filename:=/home/administrator/cartograph/b2-2016-04-27-12-31-41.bag

其中cartograph_test.pbstream 是我们上一个步骤中生成的一个地图文件,bag_filename:表示的是当前输入的激光雷达的数据

cartograph_数据集2D定位增量式更新地图

其中定位数据是输出在TF坐标系中的。通过base_link 和 odom 之间的tf 变换输出结果,可通过以下命令打印

rostopic echo /tf

2.3 小结

  • 2D demo包中没有提供里程计数据,只有激光数据和IMU 数据,输出以下3个topic
/horizontal_laser_2d
/imu
/vertical_laser_2d
  • 2D demo 运行的条件是需要已知base_link到 imu_link, horizontal_laser_link 和 vertical_laser_link(2D demo中只使用了水平激光数据)的变换数据。这个关系通过 cartographer_ros/cartographer_ros/urdf/backpack_2d.urdf 目录下的urdf文件提供

在这里插入图片描述

3 在小车上实现2D室外建图

这里我们是参考demo的历程来配置参数文件的,这里主要需要注意lua文件中的几个坐标系的配置。

  • a) 在只使用激光雷达的时候(tracking_frame=”laser”, publish_frame=”laser”)。 如果你自己发布有TF树,提供了激光雷达到base_link之间的变换,把这两个参数设置为base_link也是可以的。
  • b) 使用里程计+激光雷达时(tracking_frame=”base_link”, publish_frame=”odom”)
  • c) 使用IMU+激光+里程计时(tracking_frame=”imu_link”, publish_frame=”odom”)

注意: 每次修改cartographer工作空间里面的文件都需要重新编译

catkin_make_isolated --install --use-ninja

其余参数只要参考demo里面的进行配置就可以了,我所使用的launch文件(” mickx4_cartographer.launch”)和lua文件(“mickx4.lua”)配置如下:
在 mick_navigation/launch目录下新建launch 文件:
mickx4_carto_2D.launch

<?xml version="1.0"?>
<launch>
  <arg name="rviz" default="true"/>
   <!-- chassis -->
  <include file="$(find mick_bringup)/launch/mickx4_bringup_v2.launch" />

   <!-- imu -->
  <include file="$(find imu_driver)/launch/wit_imu.launch" />

   <!-- urdf -->
  <include file="$(find mick_description)/launch/state_publisher.launch" />

   <!-- rslidar -->
  <include file="$(find rslidar_pointcloud)/launch/rs_lidar_16.launch" />
 
   <!-- pointcloud_to_laserscan -->
  <include file="$(find pointcloud_to_laserscan)/launch/rslidar.launch" />

 <node name="cartographer_node" pkg="cartographer_ros"
      type="cartographer_node" args="
          -configuration_directory $(find cartographer_ros)/configuration_files
          -configuration_basename mickx4_mapping.lua"
      output="screen">
    <remap from="scan" to="/scan" />
    <remap from="odom" to="/odom" />
  </node>
 
  <node name="cartographer_occupancy_grid_node" pkg="cartographer_ros"
      type="cartographer_occupancy_grid_node" args="-resolution 0.05" />

 <!-- RViz -->
  <node if="$(arg rviz)" pkg="rviz" type="rviz" name="$(anon rviz)" respawn="false" output="screen" args="-d $(find mick_navigation)/rviz/mickx4_cartographer.rviz" />

</launch>

mickx4.lua

include "map_builder.lua"
include "trajectory_builder.lua"

options = {
  map_builder = MAP_BUILDER,
  trajectory_builder = TRAJECTORY_BUILDER,
  map_frame = "map",
  tracking_frame = "rslidar",
  published_frame = "base_link",
  odom_frame = "odom",
  provide_odom_frame = true, --算法内部提供里程计
  publish_frame_projected_to_2d = false,
  use_odometry = false, --使用里程计
  use_nav_sat = false,
  use_landmarks = false,

  num_laser_scans = 1,
  num_multi_echo_laser_scans = 0,
  num_subdivisions_per_laser_scan = 1,
  num_point_clouds = 0,
  lookup_transform_timeout_sec = 0.2,
  submap_publish_period_sec = 0.3,
  pose_publish_period_sec = 5e-3,
  trajectory_publish_period_sec = 30e-3,
  rangefinder_sampling_ratio = 1.,
  odometry_sampling_ratio = 1.,
  fixed_frame_pose_sampling_ratio = 1.,
  imu_sampling_ratio = 1.,
  landmarks_sampling_ratio = 1.,
}

MAP_BUILDER.use_trajectory_builder_2d = true

TRAJECTORY_BUILDER_2D.submaps.num_range_data = 35
TRAJECTORY_BUILDER_2D.min_range = 0.3
TRAJECTORY_BUILDER_2D.max_range = 8.
TRAJECTORY_BUILDER_2D.missing_data_ray_length = 1.
TRAJECTORY_BUILDER_2D.use_imu_data = false
TRAJECTORY_BUILDER_2D.imu_gravity_time_constant = 9.8 
TRAJECTORY_BUILDER_2D.use_online_correlative_scan_matching = true
TRAJECTORY_BUILDER_2D.real_time_correlative_scan_matcher.linear_search_window = 0.1
TRAJECTORY_BUILDER_2D.real_time_correlative_scan_matcher.translation_delta_cost_weight = 10.
TRAJECTORY_BUILDER_2D.real_time_correlative_scan_matcher.rotation_delta_cost_weight = 1e-1
POSE_GRAPH.optimization_problem.huber_scale = 1e2
POSE_GRAPH.optimize_every_n_nodes = 35
POSE_GRAPH.constraint_builder.min_score = 0.65

return options

lua文件的参数说明可以参考博客(cartographer系列—lua配置文件[6])和官方说明[7], 这里我将常用的几个参数罗列了过来

  • map_frame:一般为“map”.用来发布submap的ROS帧ID.
  • tracking_frame :SLAM算法要跟踪的ROS 帧ID.
  • published_frame :用来发布pose的帧ID.?
  • odom_frame: 是否使用算法内部通过匹配sacn计算的里程计
  • provide_odom_frame:如果为true,the local, non-loop-closed, continuous pose将会在map_frame里以odom_frame发布?
  • publish_frame_projected_to_2d:如果为true,则已经发布的pose将会被完全成2D的pose,没有roll,pitch或者z-offset?
  • use_odometry:如果为true,需要外部提供里程计信息,并话题/odom会订阅nav_msgs/Odometry类型的消息

下图是一个在室外的演示视频

MickX4 Cartographer 2D建图

(感谢师弟(图契图卡)、师妹帮忙录视频)

4 在小车上基于已有2D地图定位

接下来我们使用建好的地图进行定位,同时进行增量式更新地图
mickx4_cartographer_localization.launch

<launch>
   <!-- chassis -->
  <include file="$(find mick_bringup)/launch/mickx4_bringup.launch" />

   <!-- imu -->
  <include file="$(find imu_driver)/launch/wit_imu.launch" />

   <!-- urdf -->
  <include file="$(find mick_description)/launch/state_publisher.launch" />

   <!-- rslidar -->
  <include file="$(find rslidar_pointcloud)/launch/rs_lidar_16.launch" />
 
   <!-- pointcloud_to_laserscan -->
  <include file="$(find pointcloud_to_laserscan)/launch/rslidar.launch" />
   
   <node name="cartographer_node" pkg="cartographer_ros"
      type="cartographer_node" args="
          -configuration_directory $(find cartographer_ros)/configuration_files
          -configuration_basename mickx4_localization.lua
	  -load_state_filename $(arg load_state_filename)"
      output="screen">
    <remap from="echoes" to="/scan" />
  </node>
 

  <node name="cartographer_occupancy_grid_node" pkg="cartographer_ros"
      type="cartographer_occupancy_grid_node" args="-resolution 0.05" />

  <node name="rviz" pkg="rviz" type="rviz" required="true"
      args="-d $(find cartographer_ros)/configuration_files/demo_2d.rviz" />
 
</launch>

mickx4_localization.lua 定位模式下的lua配置文件,只是在建图的基础上增加了两个配置参数

include "mickx4.lua"

TRAJECTORY_BUILDER.pure_localization = true
POSE_GRAPH.optimize_every_n_nodes = 20

return options

MickX4 基于已有2D地图定位

在其他小车上的定位&增量更新地图视频:

kobuki_Cartograph_定位&增量式建图

5 大范围场景2D建图

总体上来讲cartographer要比gmapping稳定鲁棒一点,cartographer支持接入IMU和里程计,而且也支持支持地图在线更新和重定位功能。

cartographer 室外大范围2D建图

经过两次建图测试也发现四轮差速底盘的里程计确实有待提升,首先应当是接入IMU数据,直接设置航向角为IMU的融合后的yaw角,这样可以减轻航向角的误差,其次轮式里程计的误差需对轮子进行简单的标定,以提升里程计的精度。

参考资料

[1] https://www.cnblogs.com/hitcm/p/5939507.html
[2] https://blog.csdn.net/crp997576280/article/details/103279649
[3] https://github.com/cartographer-project/cartographer_ros
[4] https://github.com/cartographer-project/cartographer
[5] https://github.com/ceres-solver/ceres-solver
[6] https://blog.csdn.net/weixin_40863346/article/details/89675041
[7] https://google-cartographer-ros.readthedocs.io/en/latest/configuration.html

上一篇:开源自主导航小车MickX4(五)gmapping建图

下一篇:开源自主导航小车MickX4(七)cartographer 室外3D建图
*
欢迎大家点赞在评论区交流讨论(cenruping@vip.qq.com) O(∩_∩)O*

或者加群交流(1149897304)
在这里插入图片描述

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

开源自主导航小车MickX4(六)cartographer 室外2D建图 的相关文章

  • Python自动化备份系统及网站

    随着目前IT迅猛的发展 xff0c 自动化运维对于Linux运维人员也越来越重要 xff0c 传统的运维方式靠大量的人力 xff0c 现在也逐渐转向自动化运维 xff0c 我们常见的跟自动化有关的软件有哪些呢 今天我们来简单列举一下 xff
  • AI 黑箱难题怎么破?基于神经网络模型的算法使机器学习透明化

    编者按 xff1a 人们可以训练人工智能 xff08 AI xff09 和机器人完成任务 xff0c 但整个过程在黑箱中运作 我们并不知道 AI 和机器人是如何决策的 一家名为 OptimizingMind 的初创公司想要解决这个问题 这篇
  • 技术面试介绍

    内容 xff1a 1 技术面试的形式 2 技术面试的大致环节 3 技术面试应具备的素质 4 技术面试的经验建议 参考 xff1a 剑指offer 第二版 第一章 Java程序员面试笔试宝典 第一章 第二章 1 技术面试的形式 技术面试大致上
  • 今天是 Java 诞生日,Java 24 岁了!

    2019独角兽企业重金招聘Python工程师标准 gt gt gt 今天是 Java 诞生日 xff0c Java 今年 24 岁了 xff0c 比栈长还年轻 还有得搞 xff0c 别慌 xff01 作为一名Java语言的学习者 xff0c
  • [原创]求两个经纬度之间的距离

    经常要根据两个经纬度值 求它们之间的距离 delphi实现代码 xff1a 求两个经纬度之间的距离 function Distince const lon1 lat1 lon2 lat2 double double var alpha1 a
  • cmake的命令execute_process

    execute process COMMAND lt cmd1 gt args1 COMMAND lt cmd2 gt args2 WORKING DIRECTORY lt directory gt TIMEOUT lt seconds g
  • php 输出数组内容_php数组怎么输出

    PHP数组的输出方式有两种 xff1a print r 函数和var dump 函数 1 print r 利用 print r 函数可以打印输出整个数组内容及结构 xff0c 按照一定格式显示键和元素 注意 print r 函数不仅是只用于
  • VNC常用操作及常见问题解决办法汇总

    VNC登录用户缺省是root xff0c 但在安装oracle时必须用oracle用户的身份登录 xff0c 下面我们就以oracle为例说明如何配置VNC xff0c 从而可以使用不同的用户登录到主机 步骤描述如下 xff1a 步骤一 x
  • Xsens 在ROS环境下读取数据

    Xsens 在ROS环境下读取数据 1 Xsens 参数配置2 Xsens ROS环境下读取数据2 1 安装ROS二进制包2 2 Xsens修改rule文件2 3 启动节点 参考资料 1 Xsens 参数配置 软件可以在官网上进行下载 1
  • 交换机***查询篇

    如何确定 类型 当设备遭受 时 xff0c 通常伴随着如下现象 xff1a 用户无法获取ARP 用户上线成功率较低 用户无法访问网络 l严重时可能导致设备上所有用户都无法正常访问网络 当大量用户或固定某个端口下的所有用户出现上述现象时 xf
  • tomcat部署war包访问显示404

    在eclipse导出的maven项目war包 xff0c 放到tomcat下启动 xff0c 顺利启动且无报错 xff0c 但是浏览器访问该项目时显示404 原因 xff1a 未在server xml中配置这几个项目 xff0c 导致服务器
  • Pycharm: 代码跳转如何回退 (小技巧)

    背景 玩Python已经有段时间了 一般都是通过vim和Pycharm来开发 真心觉得这两个是神器 Vim神器暂且不说 今天来分享Pycharm的一个小技巧 用Pycharm的童鞋都知道 它有个跳转的功能 可以让我们在想深入了解某个函数 库
  • 最近沉迷美女图片无法自拔,所以我决定用PHP扒海量妹子图

    为什么80 的码农都做不了架构师 xff1f gt gt gt 学习PHP 43 MySQL制作WEB应用有一阵子了 xff0c 没有上过学所以只能学习到CURD的地步 最近对国产美女图片 私房写真特别感兴趣 xff0c 什么周妍希 刘飞儿
  • elementui的loading啊。。。踩坑了。。。

    为什么80 的码农都做不了架构师 xff1f gt gt gt 总结如下 xff1a 1 elementui的loading其实是fullscreen才是全局唯一 xff0c 不然会Loading service一次创建一次实例 2 ele
  • a padding to disable MSIE and Chrome friendly error page 填充禁用MSIE铬友好的错误页面

    一个网页在用户登录后 xff0c 应该自动刷新后 xff0c 显示登录用户信息 xff0c 但是没有自动刷新反而报错 xff0c 手动刷新后显示正常 显示如下 xff1a 检查代码 xff1a 可能原因 xff1a 文件读写权限 权限和用户
  • 边缘检测:Canny算子,Sobel算子,Laplace算子

    1 canny算子 Canny边缘检测算子是John F Canny于 1986 年开发出来的一个多级边缘检测 算法 更为重要的是 Canny 创立了边缘检测计算理论 xff08 Computational theory ofedge de
  • ProtocolException : 已超过传入消息(65536)的最大消息大小配额。

    SilverLight调用WCF xff0c 提交的是一个List lt Linq2SqlEntity gt xff1b 当List中数据量不大的时候 xff0c 不会报错 xff1b 当数据量稍微大一点儿 xff0c 就会出现这个错误 发
  • 最简单的限制edit输入

    procedure TForm1 Edt NumKeyPress Sender TObject var Key Char begin If not key in 48 57 then begin key 61 0 ShowMessage 3
  • 构建安全的数据访问-异常管理(八)

    异常条件可能会由配置错误 代码中的错误或恶意输入引起 如果没有正确的异常管理 xff0c 这些条件可能会透露有关数据源位置和特性的敏感信息 xff0c 以及有价值的连接详细信息 下面的建议适用于数据访问代码 xff1a 捕获和记录 ADO
  • 开源自主导航小车MickX4(四)底盘URDF模型

    开源自主导航小车MickX4 xff08 四 xff09 底盘URDF模型 1 底盘URDF模型1 1 URFD语法1 2 基于URDF语法绘制机器人1 3 URDF文件加载stl模型 2 绘制机器人URDF模型参考资料 1 底盘URDF模

随机推荐