ROS局部路径规划器插件teb_local_planner流程梳理(下)

news2025/1/19 2:57:00

在这里插入图片描述


   在我之前的文章《ROS导航包Navigation中的 Movebase节点路径规划相关流程梳理》中已经介绍过Move_base节点调用局部路径规划器插件的接口函数是computeVelocityCommands,本部分来,我们从这个函数入手梳理teb_local_planner功能包的工作流程。

   ☆注:因篇幅较长,本部分内容分成了上和下两篇文章,在上篇中我们已经梳理完成了规划前的准备工作,本文是对上篇的延续,在阅读本文之前,确保已经阅读完上篇的内容,以此来更好的对整个流程进行理解。

   上篇文章链接如下:

   《ROS局部路径规划器插件teb_local_planner流程梳理(上)》


   2、从里程计读取当前速度,作为起点处的速度

setVelocityStart(*start_vel);

   3、根据参数free_goal_vel设定目标点处速度,若free_goal_vel为True,则允许到达目标点处速度不为0,为false,则到达目标点处的速度要为0。

   4、调用optimizeTEB函数,根据设定的teb约束构建图,然后利用g2o优化图,也就是TEB算法核心中的核心部分,接下来,我们详细对该函数进行介绍。

optimizeTEB(cfg_->optim.no_inner_iterations, cfg_->optim.no_outer_iterations);

   (1)根据外部参数 teb_autosize的值决定是否再优化一下局部路径点,若为True,则进行优化,调用函数autoResize进行优化

 teb_.autoResize(cfg_->trajectory.dt_ref, cfg_->trajectory.dt_hysteresis, cfg_->trajectory.min_samples, cfg_->trajectory.max_samples, fast_mode);

   还记得我们在上面对plan函数介绍时的第(3)步中我们根据局部路径点中相邻两点的距离和最大线速度和角速度估计出了该两点间的运动时间,在本步中我们需要对两点间的运动时间间隔进行检查,如果某个时间间隔大于参考时间间隔加上时间滞后,且采样数小于最大采样数,则在此处插入一个新的状态;如果时间间隔小于参考时间间隔减去时间滞后,并且采样数大于最小采样数,则删除此处的状态。

   函数在进行这些插入和删除操作后会调整插入和删除点之间的时间间隔。如果fast_mode为真,则会在一次迭代中只进行一次调整,即在本次调用autoResize函数中,调整过一个点后,就返回,否则将进行多次循环,直到没有点需要调整或者达到设定的最大循环次数100才返回。

   上述过程中的参考时间间隔和时间滞后分别由外部参数 dt_ref和dt_hysteresis决定,最大采样点和最小采样点由外部参数 max_samples和min_samples决定,fast_mode由外部参数 include_dynamic_obstacles取反后得到


   (2)调用buildGraph函数进行图的构建,包括添加顶点、添加约束等过程

// 调用buildGraph函数构建图,并根据weight_multiplier进行权重调整。如果buildGraph函数执行失败,则它将清空图,并返回false。
success = buildGraph(weight_multiplier);

   ①、调用AddTEBVertices函数为图添加顶点,图的顶点包含姿态和运动时间两部分信息,将局部路径点及运动时间添加到优化器中,以便后续的约束关系可以被添加到这些顶点之间。

 AddTEBVertices();

   ②、为图添加边

   首先添加障碍物边,若外部参数 egacy_obstacle_association值为True,则调用AddEdgesObstaclesLegacy函数,采用旧的障碍物关联策略,对于每个障碍,找到最近的TEB路径点,值为false,则调用AddEdgesObstacles函数,采用新的障碍物关联策略,对于每个TEB路径点,仅找到相关"障碍

   其中AddEdgesObstacles函数的具体实现如下:首先判断是否进行了障碍物膨胀,之后定义了一个创建边的函数。之后,遍历每一个局部路径点,找到离该路径点距离小于阈值的障碍物,以及左侧和右侧最近的障碍物,构建EdgeObstacle对象,作为图的障碍物边。边误差的计算为该路径点到障碍物的距离,再过一个激活函数。

 AddEdgesObstaclesLegacy(weight_multiplier);
 AddEdgesObstacles(weight_multiplier);

   然后根据外部参数 include_dynamic_obstacles判断是否启用了动态障碍物,若启用了,则调用AddEdgesDynamicObstacles函数,添加动态障碍物边。

AddEdgesDynamicObstacles();

   然后依次调用如下函数,分别添加经过点的边、速度边、加速度边、最优时间边、最短路径边。

   其中添加经过点边的AddEdgesViaPoints函数会先遍历每一个路径点,计算与当前路径点最近的坐标点,构建路径点边,类型为 EdgeViaPoint,边的误差计算就是欧氏距离。

  AddEdgesViaPoints(); //添加经过点的边。
  
  AddEdgesVelocity(); //添加速度边。
  
  AddEdgesAcceleration(); //添加加速度边

  AddEdgesTimeOptimal(); //添加最优时间边。

  AddEdgesShortestPath(); //添加最短路径边。

   然后根据外部参数 min_turning_radius和weight_kinematics_turning_radius判断机器人的模型,若这两个参数任意一个为0,则认为当前机器人为差分驱动的机器人,即最小转弯半径为0,此时调用AddEdgesKinematicsDiffDrive函数添加差分驱动的边,若这两个参数都不为0,则认为是汽车模型,调用AddEdgesKinematicsCarlike函数添加汽车模型的边,此目的是使得生成的轨迹满足运动学约束。

AddEdgesKinematicsDiffDrive()
AddEdgesKinematicsCarlike()

   然后调用AddEdgesPreferRotDir函数添加首选旋转方向的边,

AddEdgesPreferRotDir(); //添加首选旋转方向的边。

   如果权重速度障碍物比weight_velocity_obstacle_ratio大于0,则调用AddEdgesVelocityObstacleRatio函数,添加速度障碍物比边

AddEdgesVelocityObstacleRatio()

   (3)调用optimizeGraph函数对图进行优化,首先设置优化器的详细信息并初始化优化器,并按照内循环次数参数 no_inner_iterations调用optimize函数进行具体的优化。

  // //设置优化器的详细信息并初始化优化器。
  optimizer_->setVerbose(cfg_->optim.optimization_verbose); 
  optimizer_->initializeOptimization();
  int iter = optimizer_->optimize(no_iterations); //进行指定次数的优化迭代。

   (4)清空图,按照外部参数设定的外循环次数 no_outer_iterations,循环执行上述(1)~(3)步

   (5)若参数compute_cost_afterwards为真,则在最后一次外循环迭代后调用computeCurrentCost函数计算当前的代价Cost,包含时间成本及障碍物成本,最后返回当前的总成本,缺省调用optimizeTEB函数时,compute_cost_afterwards的值默认为false。


   至此,optimizeTEB函数就执行完成了,也代表着plan函数执行完成了,到这里我们得到了优化后的路径或者说轨迹,接下来就要根据该轨迹计算每个姿态点处的线速度和角速度。


   三、计算下发给下位机的速度指令

   我们接着回到computeVelocityCommands函数中,在调用plan函数后,得到了优化后的轨迹,接下来需要进行一些准备工作,然后根据轨迹中的位姿点及时间差计算运动指令,即线速度和角速度

   1、根据外部参数 is_footprint_dynamic决定是否更新机器人的外形信息以及从机器人中心到其外形边缘顶点的最小和最大距离,若参数is_footprint_dynamic为真,则认为机器人的外形是变化的需要进行此步,否则不进行此步。

   2、调用isTrajectoryFeasible函数检查优化后得到的路径是否可行,全局路径规划得到的路径是确认可以到达终点的,经过局部路径规划优化后的轨迹会与全局路径规划有所区别,同时还有上面的可能存在的机器人模型的变化,所以对于优化后的轨迹我们需要重新判断它的可行性,根据外部参数 feasibility_check_no_poses的值决定检查轨迹中路径点的个数,比如其值为4,则检测轨迹中前4个路径点处是否与障碍物相交,需要注意的是,会把机器人的外形轮廓放置到该路径点处进行碰撞检测,而不是仅仅对该点进行检测,若有碰撞,则返回可行性检测未通过,将线速度和角速度设置为0,重置规划器,开始新的规划。

   此外,同时还会判断局部路径上每个点之间的距离以及朝向角度差,这个距离不能超过机器人的长度,如果超过了机器人的长度则点上的模型不能完全覆盖机器人路径,可能会存在说两个姿态点处没有碰撞,但是两点间的路径之间存在障碍物的问题,这样仅仅对点进行判断是不能完全保证路径可行的。此时,要对路径按照机器人长度进行差值,直到模型覆盖整条路径为止。角度的问题也是一样的。

 bool feasible = planner_->isTrajectoryFeasible(costmap_model_.get(), footprint_spec_, robot_inscribed_radius_, robot_circumscribed_radius, cfg_.trajectory.feasibility_check_no_poses);

   3、可行性检测通过后,调用getVelocityCommand函数计算速度指令,也是本部分的核心函数

getVelocityCommand(cmd_vel.twist.linear.x, cmd_vel.twist.linear.y, cmd_vel.twist.angular.z, cfg_.trajectory.control_look_ahead_poses)

   在该函数中首先检查轨迹中路径点的个数若小于2个则将速度设为0,报错返回,大于等于2,则继续进行,从局部路径点容器中依次累加两点间的时间差直至大于时间分辨率 dt_ref * control_look_ahead_poses,至此找到了局部路径点容器中和起点之间时间差大于dt_ref * control_look_ahead_poses的点,该过程还会受参数 control_look_ahead_poses(默认为1)的影响,这个参数决定了最多累加几次,默认为1时,找到的点其实就是局部路径点中起点和其后面的相邻点,无论他们的时间差是否大于dt_ref,因为只允许累加1次。其实这个参数可以理解成控制前瞻,即更倾向于选择当前点与之后的第几个点来计算当前的速度指令。

   然后,根据这两个点的位姿信息、时间差、当前的速度信息,调用extractVelocity函数,计算并更新速度信息,对于非完整约束的类车型机器人,其y轴速度为0,x轴速度为两点间距离除以时间差,若为完整约束的机器人,即y轴上的速度可以不为0在,则x轴与y轴速度分别为两点在x轴与y轴的距离差除以时间差,角速度为姿态角差除以时间差。返回true

extractVelocity(teb_.Pose(0), teb_.Pose(look_ahead_poses), dt, vx, vy, omega);

   4、速度限幅,如果优化结果违反速度约束(可能是由于软约束引起的)。则根据参数use_proportional_saturation进行处理,若该参数为默认值false,则对x、y轴速度及角速度分别按各自的比例进行处理,效果等效于若超过其对应的最大值,则限幅为其最大值。

   若use_proportional_saturation的值为True,则按同比例进行处理,比如当前x,y的速度及角速度分别为2、1、1,其对应最大值分别为 1.5、1.5、0.5,y轴速度未超过最大值,比例设为1,x轴速度超过最大值比例取为最大速度/当前速度,即1.5/2=0.75,同理角度比例为 0.5/1=0.5,取这三个比例的最小值,为共同的比例,即min(1,0.75,0.5)=0.5,则将当前速度都乘以该比例,由 2、1、1,变为1、0.5、0.5。

  saturateVelocity(cmd_vel.twist.linear.x, cmd_vel.twist.linear.y, cmd_vel.twist.angular.z,
                   cfg_.robot.max_vel_x, cfg_.robot.max_vel_y, cfg_.robot.max_vel_theta, cfg_.robot.max_vel_x_backwards);
                   

   5、非完整约束机器人的角速度转换

   若外部参数 cmd_angle_instead_rotvel为真,则需要将上面得到的角速度转换为舵机的转角,常用于类车型机器人(当然也可以不用),其具体过程是,先根据当前x轴的线速度除以角速度得到转向半径radius,若该半径小于最小转弯半径 min_turning_radius,则修正为最小转弯半径,然后根据外部参数车辆模型的前后轮间距 wheelbase与该转弯半径radius的比值取反三角函数即可得到需要的转角

  // 根据公式计算前轮转角并返回
  return std::atan(wheelbase / radius);

   6、将得到的线速度和角速度/角度发布出去,至此TEB的流程结束。


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

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

相关文章

进入银行科技部半年,已经丧失跳槽的能力了

大家好,我是锋哥!! 学弟分享 我是一个杭州双非的本科生,2022届毕业之后进了某银行的科技部工作,年包 20w。 当时想着在银行也算是一份安稳的工作,因此选择了给钱最多的一个,想着自己走上了金融 科技的赛…

Compose - 修饰符 Modifier

一、概念 四大使用场景: 修改外观(尺寸、样式、布局、行为)。添加额外信息(如无障碍标签)。添加交互功能(点击、滚动、拖拽、缩放)。处理用户输入。 1.1 为组合函数添加 Modifier 参数 任何一…

Linux网络编程:网络基础

文章目录: 1.协议 2.锁 3.网络层次模型 4.以太网帧和ARP协议 5.IP协议 6.UDP协议 7.TCP协议 8.BS模式和CS模式 9.网络套接字(socket) 10.网络字节序 11.IP地址转换函数 12.sockaddr地址结构 学习Linux的网络编程原则上基于:Linux的系统编程…

中大许少辉博士《乡村振兴战略下传统村落文化旅游设计》中国建筑工业出版社八一付梓。

中大许少辉博士《乡村振兴战略下传统村落文化旅游设计》中国建筑工业出版社八一付梓。

gdb调试的经验基本流程处理

一、启动调试 1、gdb启动 gdb启动非常简单,只要直接执行下面的命令: gdb exename(调试文件的名称)2、设置参数 如果需要调试的程序需要输入参数怎么办呢?有三种方法可以实现: a、在启动调试程序时使用命令参数设置 gdb --args …

Linux中启动docker 出现 ‘ Failed to start docker.service: Unit not found. ’ 错误

启动docker 出现 ‘ Failed to start docker.service: Unit not found. ’ 错误 这是因为缺少 rhel-push-plugin.socket 单元,该单元是rhel-push-plugin软件包的一部分。所以我们执行以下指令就可以成功解决: curl -sSL https://get.docker.com/ | sh 执…

搭建redis集群

前言 redis 集群分为一下几种: 【主从模式】:一般情况大多都是读多写少的情况,主从模式可以将读写分离,主库写,从库只负责读取的情况,这从如果任何一个从库宕机的情况,整个集群仍然可以提供工作…

使用GUI Guider工具开发嵌入式GUI应用(6)-切换多screen换场景

使用GUI Guider工具开发嵌入式GUI应用(6)-切换多screen换场景 本节将展示使用GUI Guider实现切换显示页面功能。 这里设计的用例是: 创建3张页面,screen_0,screen_1和screen_2。分别在每个页面上中放置一个Label(最…

仿牛客论坛项目day7|Kafka

一、阻塞队列 创建了一个生产者线程和一个消费者线程。生产者线程向队列中放入元素,消费者线程从队列中取出元素。我们可以看到,当队列为空时,消费者线程会被阻塞,直到生产者线程向队列中放入新的元素。 二、Kafka入门 发布、订阅…

亿图脑图MindMaster思维导图及亿图图示会员-超值途径

亿图脑图MindMaster思维导图及亿图图示会员 先简单看一下这两软件: MindMaster 亿图图示 丰富的社区,便捷易操作的界面,还有耐看的UI设计;要是再有点特权,真的是锦上添花~ 如果需要MindMaster思维导图或者亿图图示VIP…

Android Retrofit原理浅析

官方地址:Retrofit 原理:Retrofit 本质上是代理了OKhttp,使用代理模式,Type-Safe 类型安全 编译器把类型检查出 避免类型错误, enqueue 异步 切换线程 execute 同步 不切换线程 enqueue:Call接口定义的抽象方法 Retrofit.Create() 方法首先验证接口validateServiceInterf…

ps吸管工具用不了怎么办?

我们的办公神器ps软件,大家一定是耳熟能详的吧。Adobe photoshop是电影、视频和多媒体领域的专业人士,使用3D和动画的图形和Web设计人员,以及工程和科学领域的专业人士的理想选择。Photoshop支持宽屏显示器的新式版面、集20多个窗口于一身的d…

小数据 vs 大数据:为AI另辟蹊径的可操作数据

在人工智能背景下,您可能已听说过“大数据”这一流行语,那“小数据”这一词呢,您有听说过吗?无论您听过与否,小数据都无处不在:线上购物体验、航空公司推荐、天气预报等均依托小数据。小数据即一种采用可访…

webpack 和 ts 简单配置及使用

如何使用webpack 与 ts结合使用 新建项目 ,执行项目初始化 npm init -y会生成 {"name": "tsdemo01","version": "1.0.0","description": "","main": "index.js","scripts&…

java_免费文本翻译API_小牛翻译

目录 前言 开始集成API 纯文本翻译接口 双语对照翻译接口 指定术语翻译接口 总结 前言 网络上对百度,有道等的文本翻译API集成的文章比较多,所以集成的第一篇选择了小牛翻译的文本翻译API。 小牛翻译文本翻译API,支持388个语种&#xff0…

CrossOver2023快速在Mac和Linux系统上运行Windows软件

让您可以在 Mac 和 Linux 系统上运行 Windows 应用,不必购买 Windows 授权,不必重启系统,不必使用虚拟机。通过 CrossOver, 您可以从 dock 直接启动 Windows 应用,与您的 Mac 和 Linux 系统功能无缝集成。 无需重启 Cr…

强训第35天

选择 A 经过一个1->2 经过两个2->4 开始慢增长 4->5 5->6....9->10 到达4KB时变成慢增长,最多增长到10 D 网络号是180.80.76 但题目让向主机所在子网发广播 180 .80 .(01001101 | 111111 11)79 .255 标红的两位属于主机号所以答案为D A C 分片的组…

排名算法简介:对搜索结果进行排序的主要排名算法

一、介绍 学习排名 (LTR) 是一类监督式机器学习算法,旨在根据项目与查询的相关性对项目列表进行排序。在分类和回归等问题中的经典机器学习中,目标是根据特征向量预测单个值。LTR 算法对一组特征向量进行操作,并预测项…

【FAQ】视频集中存储EasyCVR安防监控平台接入AI分析时的通道显示问题排查

安防视频监控平台视频集中存储EasyCVR可拓展性强、视频能力灵活、部署轻快,可支持的主流标准协议有国标GB28181、RTSP/Onvif、RTMP等,以及支持厂家私有协议与SDK接入,包括海康Ehome、海大宇等设备的SDK等。 安防监控视频云存储平台EasyCVR既具…

为什么贴片电阻的阻值不规律?为了在某精度下做到阻值的全覆盖(对,但不全对),E系列电阻的算法(E12系列值不对的猜想)

1、先放上E系列的电阻表格 E12精度10%,E24精度5%,E96精度1%; 2、以精度作为条件的演算 通常我们选择电阻,要确定电阻的精度,如果以精度作为条件。 以10%精度来说:(数值少,好算&am…