1. 机器人地图的分类

地图有很多种表示方式,例如,用经纬度标识地方的世界地图,城市的地铁图,校园指引图。

第一种我们称为尺度地图(Metric Map),每一个地点都可以用坐标来表示,比如北京在东经116°23′17'',北纬39°54′27'';

第二种我们称为拓扑地图(Topological Map),每一个地点用一个点来表示,用边来连接相邻的点,即图论中的图(Graph),比如从地铁路线图中我们知道地铁红磡站与旺角东站和尖东站相连;

第三种我们称为语义地图(Semantic Map),其中每一个地点和道路都会用标签的集合来表示,例如,有人问我中山大学教学楼E栋在哪里,我会说在图书馆正门右手边靠近图书馆的一侧。

在机器人领域,尺度地图常用于定位于地图构建(Mapping)、定位(Localization)和同时定位与地图构建(Simultaneous Localization And Mapping,SLAM),拓扑地图常用于路径规划(Path Planning),而语义地图常用于人机交互(Human Robot Interaction)。

这里我们将介绍如何用机器人传感器数据绘制尺度地图。这有什么难点呢?首先也是最重要的一点,传感器数据有噪音。用激光传感器检测前方障碍物距离机器人多远,不可能检测到一个准确的数值。如果准确值是米,有时会测出1.42米,有时甚至1.35米。另外,传感器数据是本地坐标系的,而机器人要构建的是一个全局的地图。最后,机器人会运动,运动也是有噪音的。总结起来就两个字,噪音。通俗点来讲,“不准”。

在本篇文章中,我们利用“概率”这一神奇的数学武器来处理机器人Mapping的问题。

2. 占据栅格地图

我们首先来介绍机器人Mapping用到的的传感器,它叫做激光传感器(Laser Sensor),如下图所示:

激光传感器会向固定的方向发射激光束,发射出的激光遇到障碍物会被反射,这样就能得到激光从发射到收到的时间差,乘以速度除以二就得到了传感器到该方向上最近障碍物的距离。

这样看来,似乎利用激光传感器,机器人能够很好地完成Mapping这一任务。但是我们前面提到了,传感器数据是有噪音的。例如,假如我们在此时检测到距离障碍物4米,下一时刻检测到距离障碍物4.1米,我们是不是应该把4米和4.1米的地方都标记为障碍物?又或者怎么办呢?

为了解决这一问题,我们引入占据栅格地图(Occupancy Grid Map)的概念。

我们首先来解释这里的占据率(Occupancy)指的是什么。在通常的尺度地图中,对于一个点,它要么有(Occupied状态,下面用1来表示)障碍物,要么没有(Free状态,下面用0来表示)障碍物(旁白:那么问题来了,薛定谔状态呢?)。在占据栅格地图中,对于一个点,我们用来表示它是Free状态的概率,用来表示它是Occupied状态的概率,当然两者的和为 。两个值太多了,我们引入两者的比值来作为点的状态:

对于一个点,新来了一个测量值(Measurement,)之后我们需要更新它的状态。假设测量值来之前,该点的状态为 ,我们要更新它为: 。这种表达方式类似于条件概率,表示在z发生的条件下s的状态。

根据贝叶斯公式,我们有:

带入之后,我们得

我们对两边取对数得:

这样,含有测量值的项就只剩下了 。我们称这个比值为测量值的模型(Measurement Model),标记为。测量值的模型只有两种:,而且都是定值。

这样,如果我们用来表示位置s的状态S的话,我们的更新规则就进一步简化成了: 。其中分别表示测量值之后和之前s的状态。

另外,在没有任何测量值的初始状态下,一个点的初始状态 。

经过这样的建模,更新一个点的状态就只需要做简单的加减法了。这,就是数学的魅力。

例如,假设我们设定。那么, 一个点状态的数值越大,就表示越肯定它是Occupied状态,相反数值越小,就表示越肯定它是Free状态。

上图就展示了用两个激光传感器的数据更新地图的过程。在结果中,一个点颜色越深表示越肯定它是Free的,颜色越浅表示越肯定它是Occupied的。

3. 利用激光传感器构建占据栅格地图

前面讲到通常用激光传感器数据来构占据栅格地图,这一节我们将详细介绍其中的实现细节。具体来说,我们需要编写函数:

function myMap = occGridMapping(ranges, scanAngles, pose, param)

其中,scanAngles是一个的数组,表示激光传感器的N个激光发射方向(与机器人朝向的夹角,定值);ranges是一个K*N的数组,表示N个时间采样点激光传感器的读数(距离障碍物的距离);pose是一个3*N的数组,表示N个时间采样点机器人的位置和朝向信息(前两维为位置,第三维为朝向角度);param是一些传入的参数,param.origin是机器人的起点,param.lo_occ和param.lo_free分别是第二节中的looccu和lofree,param.max和param.min表示位置状态的阈值(超过则置为阈值边界),param.resol表示地图的分辨率,即实际地图中一米所表示的格点数目,param.size表示地图的大小。

首先,我们解决如何将真实世界中的坐标转化为栅格地图中的坐标。

考虑一维的情况:

图中x是真实世界中的坐标,i为离散化了的地图(栅格地图)中的坐标, r为一格的长度, 1/r表示分辨率,显然我们有: 。

同理,二维情况下: 。

其次,我们来计算每一条激光所检测出的障碍物和非障碍物在栅格地图中的位置。

假设机器人的状态为,激光与机器人朝向的夹角为,测量的障碍物的距离为

计算障碍物所在点的实际位置:

再计算障碍物在栅格地图中的位置 ,以及机器人在栅格地图中的位置。根据这两个坐标可以使用Bresenham算法来计算非障碍物格点的集合。

最后,利用第二节中的结论,我们使用简单的加减法不断更新格点的状态即可。

完整的Matlab代码如下:

function myMap = occGridMapping(ranges, scanAngles, pose, param)

resol = param.resol; % the number of grids for 1 meter.

myMap = zeros(param.size); % the initial map size in pixels

origin = param.origin; % the origin of the map in pixels

% Log-odd parameters

lo_occ = param.lo_occ;

lo_free = param.lo_free;

lo_max = param.lo_max;

lo_min = param.lo_min;

lidarn = size(scanAngles,1); % number of rays per timestamp

N = size(ranges,2); % number of timestamp

for i = 1:N % for each timestamp

theta = pose(3,i); % orientation of robot

% coordinate of robot in real world

x = pose(1,i);

y = pose(2,i);

% local coordinates of occupied points in real world

local_occs = [ranges(:,i).*cos(scanAngles+theta), -ranges(:,i).*sin(scanAngles+theta)];

% coordinate of robot in metric map

grid_rob = ceil(resol * [x; y]);

% calc coordinates of occupied and free points in metric map

for j=1:lidarn

real_occ = local_occs(j,:) + [x, y]; % global coordinate of occ in real world

grid_occ = ceil(resol * real_occ); % coordinate of occ in metric map

% coordinates of free in metric map (by breshnham's algorithm)

[freex, freey] = bresenham(grid_rob(1),grid_rob(2),grid_occ(1),grid_occ(2));

% convert coordinate to offset to array

free = sub2ind(size(myMap),freey+origin(2),freex+origin(1));

occ = sub2ind(size(myMap), grid_occ(2)+origin(2), grid_occ(1)+origin(1));

% update metric map

myMap(free) = myMap(free) - lo_free;

myMap(occ) = myMap(occ) + lo_occ;

end

end

% reassign value if out of range

myMap(myMap < lo_min) = lo_min;

myMap(myMap > lo_max) = lo_max;

使用课程给的数据,我们最终画出下面这样的占据栅格地图(用灰度图显示出来的):

结语:地图构建(Mapping)是机器人领域的一个重要问题,本文介绍了占据栅格地图的表示方法和利用激光传感器构建占据栅格地图的方法。但是,我们不难发现,使用的数据中机器人的位置和朝向是给定的。然而在实际的应用中,机器人不仅需要为未知环境构建地图,还要在未知环境中定位(Localization)以完成其他任务。我们发现,Mapping和Localization是相辅相成、不可分割的两部分,这就是同时定位与地图构建(SLAM)问题,机器人领域中的一个重要问题。

激光传感器构建栅格地图相关推荐

  1. velodyne+cartographer 2D构建栅格地图

    velodyne+cartographer 2D构建栅格地图   已经许久未更新博客,最近在做项目指标测试的时候,需要与单车cartographer建图进行对比实验.由于实验室的多线激光雷达是velo ...

  2. 快速构建栅格地图-MatLab

    利用Matlab快速构建栅格地图 文章目录 利用Matlab快速构建栅格地图 一.写在前面 二.地图分类 三.栅格地图优势 四.核心函数及思想 五.Matlab程序及注释 一.写在前面 ​你好,我是禅 ...

  3. matlab构建栅格地图绘图思路

    matlab构建栅格地图绘图思路 近来因研究需要,调研并思考了栅格地图的生成方法,暂时总结以备不时之需. 栅格的建立最需要注意栅格粒度的问题,即根据需要调整栅格的边长,目前有两种思路一种是固定栅格边长 ...

  4. 构建栅格地图matlab代码

    1    使用MATLAB实现地图栅格化 效果图如下所示 图1 图2 图1中的黑色栅格为障碍物栅格,白色栅格为可行走栅格(非障碍物栅格),黑色栅格 用0来表示,白色栅格用1来表示 2    栅格环境的 ...

  5. 机器人学习--栅格地图(occupancy grid map)构建(部分代码解析)

    转自: 占据栅格地图构建(Occupancy Grid Map) - 知乎 占据栅格地图构建(Occupancy Grid Map)_「小白学移动机器人」一个专注分享移动机器人相关知识的公众号!-CS ...

  6. 占据栅格地图构建(Occupancy Grid Map)

    移动机器人地图构建问题,主要以gmapping为例,讲解了地图构建的整个流程.看过前面文章的小伙伴肯定都知道,gmapping算法把SLAM问题分解成两个部分,定位问题和地图构建问题.而gmappin ...

  7. 立体栅格地图_双目立体视觉栅格地图构建方法

    双目立体视觉栅格地图构建方法 王轩 ; 叶平 ; 贾庆轩 [期刊名称] <软件> [年 ( 卷 ), 期] 2012(033)011 [摘要] 本文基于立体视觉定位技术 , 提出了基于双目 ...

  8. 基于栅格地图的路径规划(一)基于Matlab二维、三维栅格地图的构建

    基于栅格地图的路径规划(一)基于Matlab二维.三维栅格地图的构建 前言 1.二维栅格地图的创建 1.1.二维栅格地图构建原理 1.2.二维栅格地图构建例程 2.三维栅格地图的创建 2.1.三维栅格 ...

  9. 【全局规划】栅格地图

    文章目录 一.利用Matlab快速绘制栅格地图 一.利用Matlab快速绘制栅格地图 % 基于栅格地图的机器人路径规划算法 % 第1节:利用Matlab快速绘制栅格地图 clc clear close ...

  10. 栅格地图中自由区域之Bresenham算法及个人搜索算法对比

    栅格地图中的区域可以分为三种Occupied被占用(有障碍), Free自由区域(无障碍), Unknown Space未知区域. 假设图1,中黄色代表机器人,显眼的绿色代表激光雷达,红色线代表一条激 ...

最新文章

  1. DeepMind新论文:给侧面照片,AI给你脑补出正面
  2. python代码示例图形-纯干货:手把手教你用Python做数据可视化(附代码)
  3. 安装python步骤-从0到1,Python安装步骤详解(附基础知识简介)!
  4. linux完整面授视频,Linux面授实战
  5. 转载: CentOS下配置Apache
  6. 克罗谈投资策略01_期货交易中的墨菲法则
  7. 解决Docker for window与VMware虚拟机同时安装,造成虚拟机网络不通以及无法启动问题...
  8. Kubernetes Eviction Manager源码分析
  9. allegro 丝印 对齐_如何对齐丝印 - Cadence allegro PCB 教程
  10. 汉字区位码查询与算法
  11. Python 防止反编译
  12. 02-Web前端知识 day03-javaScript
  13. linux下ssh工具自动登录的实现
  14. 【qq机器人】天气查询
  15. CodeForces - 1538G Gift Set (二分)
  16. Python 数据分析之 Numpy (三)
  17. Python 之 异常值/离群值的处理
  18. 一、计算机组成与体系——软考软件设计师
  19. windows server 2008安装JDK8
  20. 树莓派4b连接蓝牙音箱/耳机播放音乐 命令行

热门文章

  1. html5做在线课件,HTML5教程:制作移动教育课件
  2. keyshot局部打光_KeyShot新手的第一次打光,无脑三点打光塑造模型质感
  3. 修马达的php源码,无刷电机控制基本原理(示例代码)
  4. 服务器安全-Dos-Deflate
  5. win10 mysql 入站规则_WIn10防火墙入站规则设置无效
  6. 智点木门软件为百闰门业解决哪些问题?
  7. 81192!请返航!
  8. linux svn图形工具,linux 下svn图形客户端smartsvn 安装
  9. Awesome Crowd Counting
  10. 不是会员不让复制粘贴?看我“三板斧”!