从零开始手写Cartographer(1): 开端
写在前面的话
我做SLAM已经三年了。读书时初学SLAM,一开始无从下手,直到读了高博士的博客,茅塞顿开,渐入佳境。后来又买了他的《视觉SLAM十四讲》,常伴手边,直至毕业。几个月前找工作的时候,他是我的面试官,与之交谈,如沐春风。那时候我就萌生出一个念头,希望追随博士的脚步,把这些年的经验和心得分享给后来人,所谓“人人为我、我为人人”,应该就是如此吧。 -- 2022.04.16
激光SLAM与Cartographer
我的水平远不及博士,只好挑自己擅长的东西说起。Cartographer是谷歌2015年发布的激光SLAM开源项目,功能完善,逻辑清晰,在学术和工程上都很具代表性。我做相关的工作已经两年了,论文和代码都读过一遍,也尝试做了一些优化和扩展,所以就拿它来做分享。
Cartographer本身是一个浩大的工程,并不利于初学者学习。本文希望根据它的设计思路,从零开始构建一个简易的2D激光SLAM工程。参考资料:
- 官网: https://google-cartographer.readthedocs.io/en/latest/
- github: https://github.com/cartographer-project/cartographer
- 论文:Real-Time Loop Closure in 2D LIDAR SLAM
这里放出一张官方的动图:

SLAM之我见
这里先说说什么是SLAM:SLAM译作同时定位与地图构建,被认为是实现真正全自主移动机器人的关键。我记得这是我硕士论文的第一句,听着确实有掉书袋的嫌疑,也确实费耳朵。从我工作经验来讲,SLAM问题姑且可以等同于:如何通过整合局部观测来建立全局地图。
激光SLAM的观测是点云,每帧点云都是对环境的局部观测;而激光SLAM的目的就是用各时刻的点云拼成全局地图。这个过程如同拼图一样:
当然,以上是我对于“基于图优化的SLAM方法”做的理解,自然有偏颇之处。这里建议初学者不必拘泥于具体概念,相信有一定项目基础后会有更深的体会。
让我们开始吧!
开发环境:Linux,推荐Ubuntu。
其实如果我们的代码是纯C++且不依赖于第三方库,那么无论何种平台(Windows/Linux/MacOS)都可以编译运行。但实际上我们经常会用到仅支持Linux的第三方库,所以我们不得不用Linux。
实际上,Linux也是非常基本的开发环境,而ubuntu是最知名的Linux发行版,有了问题也容易找到资料。如果手头没有Linux环境,可以用WSL或云服务器,这里不再详述。
对于一个崭新的ubuntu环境来说,我们通常需要安装一些基本的工具来编译:
# gcc等基本编译套件
sudo apt install build-essential
# cmake
sudo apt install cmake
然后我们创建工程目录,姑且称之为slam。然后创建src目录存放代码,build目录存放编译产物:
sudo mkdir slam slam/src slam/build
我们在src目录中写一个最简单的C++程序,保存为hello.cc:
#include <iostream>
int main() {
std::cout << "hello, slam!" << std::endl;
return 0;
}
这里我们用了Google Style作为工程规范。
我们使用cmake来编译我们的工程。需要在slam下写一个CMakeLists.txt:
# 指定最低版本
cmake_minimum_required(VERSION 2.8)
# 指定项目名
project(slam)
# 用src/hello.cc产生一个名为hello的可执行项目
add_executable(hello src/hello.cc)
此时路径树应该是这样的:
slam\
|---src\
| |--- hello.cc
|
|---build\
|
|---CMakeLists.txt
我们进入build目录,执行编译:
cd build
cmake ..
make
此时会在build下产生hello可执行文件,可以执行它:
./hello
此时如果输出"hello, slam!"字样,说明成功了。
到此为止,我们有了一个好的开始。下节我们讲述一些激光SLAM的基础概念,并做工程实现。
以上工程可以参考我的github(https://github.com/ysklab/cartographer-from-scratch)仓库,需要手动checkout到指定commit。本节的commit信息为:
start with "hello slam"
这里假设读者具备基本的git操作知识,包括clone、log、checkout等。
从零开始手写Cartographer(1): 开端的更多相关文章
- java 从零开始手写 RPC (03) 如何实现客户端调用服务端?
说明 java 从零开始手写 RPC (01) 基于 socket 实现 java 从零开始手写 RPC (02)-netty4 实现客户端和服务端 写完了客户端和服务端,那么如何实现客户端和服务端的 ...
- java 从零开始手写 RPC (04) -序列化
序列化 java 从零开始手写 RPC (01) 基于 socket 实现 java 从零开始手写 RPC (02)-netty4 实现客户端和服务端 java 从零开始手写 RPC (03) 如何实 ...
- java 从零开始手写 RPC (05) reflect 反射实现通用调用之服务端
通用调用 java 从零开始手写 RPC (01) 基于 socket 实现 java 从零开始手写 RPC (02)-netty4 实现客户端和服务端 java 从零开始手写 RPC (03) 如何 ...
- java 从零开始手写 RPC (07)-timeout 超时处理
<过时不候> 最漫长的莫过于等待 我们不可能永远等一个人 就像请求 永远等待响应 超时处理 java 从零开始手写 RPC (01) 基于 socket 实现 java 从零开始手写 RP ...
- 从零开始手写 dubbo rpc 框架
rpc rpc 是基于 netty 实现的 java rpc 框架,类似于 dubbo. 主要用于个人学习,由渐入深,理解 rpc 的底层实现原理. 前言 工作至今,接触 rpc 框架已经有很长时间. ...
- 从零开始手写 spring ioc 框架,深入学习 spring 源码
IoC Ioc 是一款 spring ioc 核心功能简化实现版本,便于学习和理解原理. 创作目的 使用 spring 很长时间,对于 spring 使用非常频繁,实际上对于源码一直没有静下心来学习过 ...
- java 从零开始手写 RPC (01) 基于 websocket 实现
RPC 解决的问题 RPC 主要是为了解决的两个问题: 解决分布式系统中,服务之间的调用问题. 远程调用时,要能够像本地调用一样方便,让调用者感知不到远程调用的逻辑. 这一节我们来学习下如何基于 we ...
- AI应用开发实战 - 手写识别应用入门
AI应用开发实战 - 手写识别应用入门 手写体识别的应用已经非常流行了,如输入法,图片中的文字识别等.但对于大多数开发人员来说,如何实现这样的一个应用,还是会感觉无从下手.本文从简单的MNIST训练出 ...
- 放弃antd table,基于React手写一个虚拟滚动的表格
缘起 标题有点夸张,并不是完全放弃antd-table,毕竟在react的生态圈里,对国人来说,比较好用的PC端组件库,也就antd了.即便经历了2018年圣诞彩蛋事件,antd的使用者也不仅不减,反 ...
随机推荐
- 华硕主板安装Ubuntu双系统无法启动解决办法
问题描述: 在安装完Ubuntu后,开机后没有启动项可以选择,而是直接进入win10系统. 解决办法: 开机进入bios,选择 高级模式 (Advance model),进入 启动(boot)选项卡, ...
- 团队Beta5
队名:观光队 链接 组长博客 作业博客 组员实践情况 王耀鑫 **过去两天完成了哪些任务 ** 文字/口头描述 学习 展示GitHub当日代码/文档签入记录 无 接下来的计划 无 **还剩下哪些任务 ...
- CTO 说了,如果发现谁用 kill -9 关闭程序就开除
关注「开源Linux」,选择"设为星标" 回复「学习」,有我为您特别筛选的学习资料~ 来源:blog.csdn.net/qq_33220089 正文 kil ...
- Java-GUI 编程之 JList、JComboBox实现列表框
目录 JList.JComboBox实现列表框 简单列表框 不强制存储列表项的ListModel和ComboBoxModel 强制存储列表项的DefaultListModel和DefaultCombo ...
- 如何基于 ZEGO SDK 实现 Android 通话质量监测
功能简介 在进行视频通话过程中,用户有时候会出现网络不好的情况,比如在进行多人视频通话或者多人唱歌时,我们需要实时显示用户的网络质量. 示例源码 参考 下载示例源码 获取源码. 相关源码请查看 &qu ...
- Python 散列表查询_进入<哈希函数>为结界的世界
1. 前言 哈希表或称为散列表,是一种常见的.使用频率非常高的数据存储方案. 哈希表属于抽象数据结构,需要开发者按哈希表数据结构的存储要求进行 API 定制,对于大部分高级语言而言,都会提供已经实现好 ...
- c++ :STL
基础知识 容器 容器就是一些模板类的集合,不同之处就是容器中封装的是数据结构 1.序列容器 主要有vector向量容器.list列表容器.deque双端队列容器 元素在容器中是无序的 2.排序容器 包 ...
- 判断数据类型(typeof&instanceof&toString)
一.数据类型 ES6规范中有7种数据类型,分别是基本类型和引用类型两大类 基本类型(简单类型.原始类型):String.Number.Boolean.Null.Undefined.Symbol 引用类 ...
- FileAPI
FileAPI ```java File类的常见方法 1.创建. boolean createNewFile(); //创建文件 boolean mkdir();创建文件夹 boolean mkdir ...
- 这篇 DNS ,写的挺水的。
试想一个问题,我们人类可以有多少种识别自己的方式?可以通过身份证来识别,可以通过社保卡号来识别,也可以通过驾驶证来识别,尽管有多种识别方式,但在特定的环境下,某种识别方法会比其他方法更为适合.因特网上 ...