欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

从零开始做自动驾驶定位(九): 建图系统结构优化

程序员文章站 2022-07-12 10:07:02
...

文纯属转载,并认真学习一遍,感谢大佬分享!

注释:文中蓝色文本是自己加上去的

本文章配套源代码地址:https://github.com/Little-Potato-1990/localization_in_auto_driving

测试数据:https://pan.baidu.com/s/1TyXbifoTHubu3zt4jZ90Wg提取码: n9ys

本篇文章对应的代码Tag为 9.0

代码在后续可能会有调整,如和文章有出入,以实际代码为准

===========================================================================================

一、概述

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

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

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

所以本篇文

二、设计思路

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

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上的显示中,当前点云、位姿与全局地图自然就对应不上,所以要做一次投影。投影所使用的坐标就是看优化时对之前的位姿做了多少修正,这个修正量就是要投影的转换量。

 

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

上一篇:从零开始做自动驾驶定位(八): 点云畸变补偿

下一篇:从零开始做自动驾驶定位(十): 后端优化

 

 

 

相关标签: 激光SLAM