机器人算法——costmap膨胀层InflationLayer

如下图是更新地图膨胀

void InflationLayer::updateCosts(costmap_2d::Costmap2D& master_grid, int min_i, int min_j, int max_i, int max_j)
{//用指针master_array指向主地图,并获取主地图的尺寸,确认seen_数组被正确设置。boost::unique_lock < boost::recursive_mutex > lock(*inflation_access_);if (!enabled_ || (cell_inflation_radius_ == 0))return;// make sure the inflation list is empty at the beginning of the cycle (should always be true)ROS_ASSERT_MSG(inflation_cells_.empty(), "The inflation list must be empty at the beginning of inflation");unsigned char* master_array = master_grid.getCharMap();unsigned int size_x = master_grid.getSizeInCellsX(), size_y = master_grid.getSizeInCellsY();if (seen_ == NULL) {ROS_WARN("InflationLayer::updateCosts(): seen_ array is NULL");seen_size_ = size_x * size_y;seen_ = new bool[seen_size_];}else if (seen_size_ != size_x * size_y){ROS_WARN("InflationLayer::updateCosts(): seen_ array size is wrong");delete[] seen_;seen_size_ = size_x * size_y;seen_ = new bool[seen_size_];}memset(seen_, false, size_x * size_y * sizeof(bool));// We need to include in the inflation cells outside the bounding// box min_i...max_j, by the amount cell_inflation_radius_.  Cells// up to that distance outside the box can still influence the costs// stored in cells inside the box.//边际膨胀min_i -= cell_inflation_radius_;min_j -= cell_inflation_radius_;max_i += cell_inflation_radius_;max_j += cell_inflation_radius_;min_i = std::max(0, min_i);min_j = std::max(0, min_j);max_i = std::min(int(size_x), max_i);max_j = std::min(int(size_y), max_j);// Inflation list; we append cells to visit in a list associated with its distance to the nearest obstacle// We use a map<distance, list> to emulate the priority queue used before, with a notable performance boost//接下来遍历bound中的cell,找到cost为LETHAL_OBSTACLE,即障碍物cell,将其以CellData形式放进inflation_cells_[0.0]中,inflation_cells_的定义如下://std::map<double, std::vector > inflation_cells_;//它是一个映射,由浮点数→CellData数组,CellData这个类定义在inflation_layer.h中,//是专门用来记录当前cell的索引和与它最近的障碍物的索引的。这个映射以距离障碍物的距离为标准给bound内的cell归类,为膨胀做准备;//自然距离为0对应的cell即障碍物cell本身,//目前得到的inflation_cells_只包括障碍物本身。// Start with lethal obstacles: by definition distance is 0.0std::vector<CellData>& obs_bin = inflation_cells_[0.0];for (int j = min_j; j < max_j; j++){for (int i = min_i; i < max_i; i++){int index = master_grid.getIndex(i, j);unsigned char cost = master_array[index];if (cost == LETHAL_OBSTACLE){obs_bin.push_back(CellData(index, i, j, i, j));}}}// Process cells by increasing distance; new cells are appended to the corresponding distance bin, so they// can overtake previously inserted but farther away cells//通过增加距离处理细胞;新的单元格被附加到相应的距离仓中,因此它们//可以超越先前插入但距离更远的单元格std::map<double, std::vector<CellData> >::iterator bin;for (bin = inflation_cells_.begin(); bin != inflation_cells_.end(); ++bin){for (int i = 0; i < bin->second.size(); ++i){// process all cells at distance dist_bin.firstconst CellData& cell = bin->second[i];unsigned int index = cell.index_;// ignore if already visitedif (seen_[index]){continue;}seen_[index] = true;unsigned int mx = cell.x_;unsigned int my = cell.y_;unsigned int sx = cell.src_x_;unsigned int sy = cell.src_y_;// assign the cost associated with the distance from an obstacle to the cellunsigned char cost = costLookup(mx, my, sx, sy);unsigned char old_cost = master_array[index];if (old_cost == NO_INFORMATION && (inflate_unknown_ ? (cost > FREE_SPACE) : (cost >= INSCRIBED_INFLATED_OBSTACLE)))master_array[index] = cost;elsemaster_array[index] = std::max(old_cost, cost);// attempt to put the neighbors of the current cell onto the inflation listif (mx > 0)enqueue(index - 1, mx - 1, my, sx, sy);if (my > 0)enqueue(index - size_x, mx, my - 1, sx, sy);if (mx < size_x - 1)enqueue(index + 1, mx + 1, my, sx, sy);if (my < size_y - 1)enqueue(index + size_x, mx, my + 1, sx, sy);}}inflation_cells_.clear();
}

InflationLayer没有自身的栅格地图要维护,直接在主地图上进行操作,它根据膨胀参数设置用来膨胀的“参考矩阵”,并在主地图上从障碍物出发,不断传播更新,完成对整个地图障碍的膨胀。
不断的从障碍物周围”上下左右“开始膨胀,并且根据参考矩阵去计算不同距离的栅格权重。

 // assign the cost associated with the distance from an obstacle to the cellunsigned char cost = costLookup(mx, my, sx, sy);unsigned char old_cost = master_array[index];if (old_cost == NO_INFORMATION && (inflate_unknown_ ? (cost > FREE_SPACE) : (cost >= INSCRIBED_INFLATED_OBSTACLE)))master_array[index] = cost;elsemaster_array[index] = std::max(old_cost, cost);

unsigned char cost = costLookup(mx, my, sx, sy);为参考矩阵计算, master_array[index] = cost;为当前cell栅格赋值。

 // attempt to put the neighbors of the current cell onto the inflation listif (mx > 0)enqueue(index - 1, mx - 1, my, sx, sy);if (my > 0)enqueue(index - size_x, mx, my - 1, sx, sy);if (mx < size_x - 1)enqueue(index + 1, mx + 1, my, sx, sy);if (my < size_y - 1)enqueue(index + size_x, mx, my + 1, sx, sy);}

这是考察下一个点,上下左右,如果在障碍物半径内,则继续循环到上述的“ unsigned char cost = costLookup(mx, my, sx, sy);为参考矩阵计算, master_array[index] = cost;为当前cell栅格赋值。”

如此循环即可,类似于A*算法的膨胀。

/**
*@brief给定成本图中某个单元格的索引,将其放入等待障碍物膨胀的列表中
*@param grid成本图
*@param index单元格的索引
*@param mx单元格的x坐标(可以根据索引计算,但可以节省存储时间)
*@param my单元格的y坐标(可以根据索引计算,但可以节省存储时间)
*@param src_x障碍点通货膨胀的x指数开始于
*@param src_y障碍点通货膨胀的y指数始于
*/
inline void InflationLayer::enqueue(unsigned int index, unsigned int mx, unsigned int my,unsigned int src_x, unsigned int src_y)
{if (!seen_[index]){// we compute our distance table one cell further than the inflation radius dictates so we can make the check belowdouble distance = distanceLookup(mx, my, src_x, src_y);// we only want to put the cell in the list if it is within the inflation radius of the obstacle pointif (distance > cell_inflation_radius_)return;// push the cell data onto the inflation list and markinflation_cells_[distance].push_back(CellData(index, mx, my, src_x, src_y));}
}
 double distance = distanceLookup(mx, my, src_x, src_y);

使用计算考察点和障碍物的距离,然后判断是否在膨胀半径内,如果是,则保存到nflation_cells_中用于下一次循环中。

Reference

  1. 机器人控制—代价地图Costmap的层概述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/200112.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

pthread学习遇到的问题

1.pthread_t 是个类型&#xff0c;指的是线程ID。pthread_create&#xff08;&#xff09;的时候穿地址进去&#xff0c;线程创建好后就会成为线程ID&#xff08;即输出型参数&#xff09; 2.pthread_self() pthread_self()获得是调用这个函数的线程ID &#xff08;我以为是…

Shell数组函数:数组(二)

关联数组 注意&#xff1a;先声明关联数组 一、定义关联数组 方法一 #一次赋一值 #数组名[索引]变量值 [rootlocalhost ~]# declare -A ass_array1 [rootlocalhost ~]# ass_array1[index1]pear [rootlocalhost ~]# ass_array1[index2]apple [rootlocalhost ~]# ass_array1[ind…

java中Random随机数使用和生成随机数的多个示例

在 Java 中&#xff0c;我们可以使用 java.util.Random 类生成伪随机数。伪随机数的特性是&#xff0c;虽然它们看起来是随机的&#xff0c;但实际上它们是由一个固定的算法生成的。只要我们提供相同的种子&#xff0c;这个算法就会生成相同的数字序列。 首先&#xff0c;我们…

详解十大经典排序算法(五):归并排序(Merge Sort)

算法原理 归并排序的核心思想是将一个大的数组分割成多个小的子数组&#xff0c;然后分别对这些子数组进行排序&#xff0c;最后将排序后的子数组合并起来&#xff0c;得到一个有序的大数组。 算法描述 归并排序&#xff08;Merge Sort&#xff09;是一种经典的排序算法&#x…

CoreDNS实战(六)-编译安装unbound插件

本文主要介绍coredns的unbound插件进行编译安装的过程及常用的配置方法。 coredns官方的unbound文档&#xff1a;unbound unbound插件的github地址&#xff1a;https://github.com/coredns/unbound 注&#xff1a;unbound插件虽然是coredns中的External Plugins&#xff0c;但…

机器学习中Fine-tuning应用实例

Fine-tuning&#xff08;微调&#xff09;是机器学习中一种常见的训练策略&#xff0c;其主要作用是在一个已经在大规模数据上预训练的模型基础上&#xff0c;通过使用特定任务的小规模数据集来进行额外的训练。Fine-tuning通常用于调整模型&#xff0c;使其适应新的任务或特定…

CMake中的include(CPack)

2023年12月5日&#xff0c;周二晚上 include(CPack) 是在 CMakeLists.txt 文件中包含 CPack 配置的指令。当在 CMakeLists.txt 文件中添加 include(CPack) 时&#xff0c;它会告诉 CMake 在构建过程中加载 CPack 的相关功能和配置。 具体来说&#xff0c;include(CPack) 会包含…

网络运维与网络安全 学习笔记2023.12.5

网络运维与网络安全 学习笔记 第三十五天 今日目标 su用户切换、sudo命令提权、部署动态Web应用 数据库安全加固、Web安全加固 网络监控基础、配置zabbix主控机、配置zabbix被控机 管理监控项、监控结果分析 su用户切换 su机制介绍及用法 Linux安全基线 指的是使Linux各项…

【Linux】24、文件系统、磁盘 IO

文章目录 一、文件系统1.1 索引节点和目录项1.2 虚拟文件系统 VFS1.3 文件系统 I/O1.5 性能观测1.5.1 容量1.5.2 缓存1.5.3 find 命令的缓存 二、磁盘 I/O2.1 通用块层2.2 I/O 栈2.3 磁盘性能指标2.3.1 磁盘 I/O 观测2.3.2 进程 I/O 观测 2.4 案例&#xff1a;找到打大量日志的…

【开源】基于Vue.js的贫困地区人口信息管理系统

文末获取源码&#xff0c;项目编号&#xff1a; S 073 。 \color{red}{文末获取源码&#xff0c;项目编号&#xff1a;S073。} 文末获取源码&#xff0c;项目编号&#xff1a;S073。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 人口信息管理模块2.2 精准扶贫管理模…

修改el-table表头样式

<style lang"scss" scoped> ::v-deep .el-table {.el-table__header-wrapper, .el-table__fixed-header-wrapper {th {word-break: break-word;background-color: #f8f8f9;color: #515a6e;height: 40px;font-size: 13px;}} } </style>

el-table全部选择和全部取消

el-table实现全部选择和全部取消 其实非常简单&#xff0c;el-table自带的都有方法toggleAllSelection()和clearSelection() 具体代码如下&#xff1a; <el-button typesuccess clickcheckAll sizesmall>全选</el-button> <el-button typesuccess clickcancel…

【苍穹外卖】——day3

文章目录 公共字段自动填充一、新增菜品二、菜品分页查询三、删除菜品分类四、修改菜品接口总结 公共字段自动填充 create_time,create_user,update_time,update_user这四个字段自动填充 步骤&#xff1a; 自定义注解AutoFill&#xff0c;用于标识需要进行公共字段自动填充的方…

设置python下载包代理

使用场景 正常网络情况下我们安装如果比较多的python包时&#xff0c;会选择使用pip install -r requirements.txt -i https://pypi.douban.com/simple --trusted-hostpypi.douban.com这种国内的镜像来加快下载速度。 但是&#xff0c;当这台被限制上网时&#xff08;公司安全…

基于ROPNet项目训练modelnet40数据集进行3d点云的配置

项目地址&#xff1a; https://github.com/zhulf0804/ROPNet 在 MVP Registration Challenge (ICCV Workshop 2021)&#xff08;ICCV Workshop 2021&#xff09;中获得了第二名。项目可以在win10环境下运行。 论文地址&#xff1a; https://arxiv.org/abs/2107.02583 网络简介…

【PID学习笔记 6 】控制系统的性能指标之二

写在前面 上文介绍了控制系统的稳态与动态、过渡过程、阶跃响应以及阶跃信号作用下过渡过程的四种形式。本文紧接上文&#xff0c;首先总结过渡过程的分类&#xff0c;然后介绍控制系统的性能评价&#xff0c;最后重点介绍控制系统性能指标中的单项指标。 一、过渡过程的分类…

int(1)和int(10)之间的区别

在Java编程语言中&#xff0c;int(1)和int(10)之间的区别主要在于它们的初始化值和类型。以下是详细的技术文档&#xff1a; 初始化值&#xff1a; int(1)表示将整数1赋值给变量&#xff0c;而int(10)表示将整数10赋值给变量。这意味着它们的初始化值不同&#xff0c;分别为1和…

Unity资源路径与读取

Unity资源路径有&#xff1a; 1、StreamingAssets&#xff1a;只读&#xff0c;一般用于存放应用程序运行时需要加载的资源文件&#xff0c;可以通过Application.streamingAssetsPath来获取。 2、PersistentDataPath&#xff1a;可读写&#xff0c;一般用于存放应用程序运行时…

java版微信小程序商城免费搭建 java版直播商城平台规划及常见的营销模式有哪些?电商源码/小程序/三级分销

涉及平台 平台管理、商家端&#xff08;PC端、手机端&#xff09;、买家平台&#xff08;H5/公众号、小程序、APP端&#xff08;IOS/Android&#xff09;、微服务平台&#xff08;业务服务&#xff09; 2. 核心架构 Spring Cloud、Spring Boot、Mybatis、Redis …

解密IIS服务器API跨域问题的终极解决方案

在当今数字化时代&#xff0c;API已成为现代应用程序的核心组件。然而&#xff0c;当你使用IIS&#xff08;Internet Information Services&#xff09;服务器提供API时&#xff0c;你可能会遇到一个常见的挑战&#xff1a;API跨域问题。这个问题经常困扰着开发人员&#xff0c…