自动驾驶定位系列教程八_ 建图系统结构优化

自动驾驶定位系列教程八: 建图系统结构优化

一、概述

我们通过前面几篇文章,已经把前端里程计模块的功能实现了,在里程计之前做了时间同步、点云畸变补偿等预处理,在里程计之后做了实时显示、轨迹精度评测等。这种“之前”和“之后”的描述已经具备了模块划分的概念,即分成了三个模块,只不过我们没有严格按照模块去管理代码。

在建图功能后面的工作里,我们要添加的就是后端优化、闭环检测这两大模块了。

这样我们就有了五个模块,模块数量增加了,就需要有一个更好的代码结构来管理。

所以本篇文章的目的就是设计这个代码结构,在我们的系列工作中起到一个承上启下的作用。

image

附赠自动驾驶学习资料和量产经验:链接

本系列教程涉及完整代码V作者获取

二、设计思路

上面已经梳理清楚模块的数量和基本功能,那么就需要对每个模块做一个封装,并把代码文件用合适的目录管理好就行。具体封装什么细节,就需要对模块做详细分析了。

1. 模块功能详解

我们再详细看一下这五个模块的具体内容。

1)数据预处理

a. 功能

  • 接收各传感器信息

  • 传感器数据时间同步

  • 点云运动畸变补偿

  • 传感器信息统一坐标系

b. 输入

  • GNSS组合导航位置、姿态、角速度、线速度等

  • 雷达点云信息

  • 雷达和IMU相对坐标系

c. 输出

  • GNSS组合导航位置、姿态

  • 畸变补偿后的点云

备注:以上信息均是经过时间同步的,时间戳已保持一致。

2)前端里程计

a. 功能

  • 根据点云计算位姿并发送

b. 输入

  • 雷达点云

c. 输出

  • 里程计累计位姿

3)后端图优化

a. 功能

  • 从里程计位姿中提取关键帧

  • 根据关键帧位姿、GNSS组合导航约束和闭环检测约束来优化位姿

b. 输入

  • 前端里程计位姿

  • GNSS组合导航位姿

  • 闭环检测相对位姿

c. 输出

  • 优化后的位姿

4)闭环检测

a. 功能

  • 根据位姿搜索附近历史帧

  • 如果有则做匹配

  • 如果匹配结果符合要求则建立约束,并发送

b. 输入

  • 关键帧对应的GNSS组合导航位姿(因为用开环位姿检测不够准确,尤其大场景)

c. 输出

  • 闭环位姿约束

5)显示模块

a. 功能

  • 根据优化后的位姿生成点云地图

  • 显示点云地图和当前帧点云

b. 输入

  • 优化后的历史帧位姿

  • 当前帧点云

  • 当前帧位姿

c. 输出

  • 按优化后的位姿投影之后的当前帧点云

  • 按优化后的位姿投影之后的当前帧位姿

  • 局部小地图点云

  • 全局大地图点云

2. 代码文件管理

通过上面一部分,我们看到第一个模块是建图功能和以后要做的定位功能共有的模块,后面四个模块是建图功能独有的模块。所以我们代码管理的基本思路就是在工程目录先建立文件夹"mapping",把后四个模块放在文件夹里,第一个模块独立在文件夹外面。

按照之前的思路,为了代码简洁,每个模块内部应该把输入输出功能与核心算法功能分开,而且node文件只是调用模块,所以每个模块就对应了三部分:

1)核心功能

负责实现该模块对应的核心功能,用类封装。放在该模块对应的文件夹下。文件名与功能模块名相同。

2)流程管理

负责该模块的输入输出,并调用核心功能的类。放在该模块对应文件夹下,为区分,文件名以"_flow"结尾。

3)node文件

控制ros循环,调用流程管理的类。我们在工程目录下新建了一个"apps"文件夹,专门用来存放各模块的node文件,为区分,各文件名字以"_node"结尾。

4)配置文件

除了数据预处理功能以外,其他功能模块都配备了配置文件,方便调试。配置文件放在工程目录里的"config"文件夹里。

完成了上面两部分的分析,就可以看各个模块的具体实现了。

三、代码实现

其实模块功能清晰了以后,代码就没太多可说的了。

前端核心算法还是那些,后端图优化部分目前只有基本流程,还没对位姿做优化,所以现在后端模块就是吃什么吐什么,把从前端接收过来的位姿再发给显示模块。

闭环模块暂时没填具体代码,等具体做到那一步的时候再写。

1. 数据预处理

节点文件:apps/data_pretreat_node.cpp

流程管理文件:data_pretreat/data_pretreat_flow.cpp

核心算法文件:data_pretreat/data_pretreat.cpp

没太多可说的,它内部的内容在以前几篇文章都介绍过了,只不过这次重新划分了文件。

2. 前端里程计

节点文件:apps/front_end_node.cpp

流程管理文件:mapping/front_end/front_end_flow.cpp

核心算法文件:mapping/front_end/front_end.cpp

配置文件:config/front_end/config.yaml

以前的代码里有前端里程计的具体实现,此处只做了减法,具体包括:

1)把全局地图生成动能去掉了,放在了其他模块里,因为计算位姿不需要全局地图

2)把关键帧存储成pcd文件的功能去掉了,放在了其他模块里,因为此处只是里程计,不包括关键帧管理

3)groun_truth和odom两个txt文件的存储功能去掉了,放在了其他模块里。因为里程计计算不需要gnss位姿,所以应该和这部分功能解耦。

3. 后端图优化

节点文件:apps/back_end_node.cpp

流程管理文件:mapping/back_end/back_end_flow.cpp

核心算法文件:mapping/back_end/back_end.cpp

配置文件:config/back_end/config.yaml

暂时没对位姿做优化,目前的功能有:

1)接收前端里程计和GNSS组合导航数据,并按时间对齐之后写在txt文件里,方便评测里程计精度

2)从10HZ的里程计位姿中识别出关键帧,并把关键帧位姿和对应的GNSS组合导航位姿发送出去

3)发送历史帧系列位姿,这个功能是为了后面添加图优化功能时使用,每次优化后,把优化后的位姿读取出来,发送给其他模块用。

4. 闭环检测功能

暂时没填写这个功能的代码

5. 显示功能

节点文件:apps/viewer_node.cpp

流程管理文件:mapping/viewer/viewer_flow.cpp

核心算法文件:mapping/viewer/viewer.cpp

配置文件:config/viewer/config.yaml

目前实现的主要功能有:

1)接收优化后的位姿,并根据这个位姿生成全局地图。

由于优化频率低,所以优化生成地图的关键帧可能要比当前全部的关键帧要少很多,具体少多少,可以在优化模块的配置文件里设置。之所以只用优化后的位姿建图,是因为没经过优化的位姿不够精确,用来拼接地图会把地图弄出很多重影,所以我们宁愿舍弃。

另一点是,程序会自动检测rviz上是否要接收地图文件,即rviz界面左侧对应的信息是否被勾选,当勾选时,才生成地图并发送。之所以要这么做是因为,地图文件比较大,生成和发送都比较耗时间,在建图过程中不一定需要实时观看全局地图,这样可以加快速度。

如果想保存最后生成的全局点云地图,可以输入下面的指令

rosservice call /save_map

它会在工程目录的slam_data/map文件夹下生成"map.pcd"文件,这就是地图文件。

当然,地图的生成目录也可以在配置文件中自己指定。

2)接收当前帧点云和位姿,按优化后的位姿投影并发送。

我们接收到的点云是预处理那一步产生的点云,位姿是优化前的里程计位姿,当后端做过图优化之后,如果不投影,那rviz上的显示中,当前点云、位姿与全局地图自然就对应不上,所以要做一次投影。投影所使用的坐标就是看优化时对之前的位姿做了多少修正,这个修正量就是要投影的转换量。

虽然图优化算法和闭环检测算法的具体实现还没加,但这一次改过的架构仍然是可以正常运行的,只不过它实现的仍是里程计功能,区别是内部实现变了,对模块做了划分。运行结果就不往这里贴了,和以前没啥区别。

完整教程联系作者获取

  • 25
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值