【recast-navigation/源码解析】findStraightPath详解以及寻路结果贴边优化
说在前面
- recast-navigation版本:1.6.0
叉积cross product
- 正常来讲,叉乘为:
∣ A ⃗ × B ⃗ ∣ = ∣ x A y A x B y B ∣ = x A ⋅ y B − x B ⋅ y A |\vec{A} \times \vec{B}|=\begin{vmatrix} x_A & y_A \\ x_B & y_B \end{vmatrix}=x_A \cdot y_B - x_B \cdot y_A ∣A×B∣= xAxByAyB =xA⋅yB−xB⋅yA - 例如:

- 但是在
recast-navigation库中:
这应该是因为x轴与正常坐标系相反/// Derives the signed xz-plane area of the triangle ABC, /// or the relationship of line AB to point C. /// @param[in] a Vertex A. [(x, y, z)] /// @param[in] b Vertex B. [(x, y, z)] /// @param[in] c Vertex C. [(x, y, z)] /// @return The signed xz-plane area of the triangle. inline float dtTriArea2D(const float* a, const float* b, const float* c) {const float abx = b[0] - a[0];const float abz = b[2] - a[2];const float acx = c[0] - a[0];const float acz = c[2] - a[2];return acx*abz - abx*acz; }
findStraightPath
-
在使用
findStraightPath前,我们通常会先使用findPath进行A*搜索,得到一个基于多边形的路径,假设findPath的结果如下图:

-
而
findStraightPath的目的则是从这几个多边形中寻找一条路径出来,接下来我们一步步的分解这个过程-
首先我们对一些计算过程中使用到的临时变量进行初始化, 主要为
portalApex、portalLeft、portalRight;float portalApex[3], portalLeft[3], portalRight[3]; dtVcopy(portalApex, closestStartPos); dtVcopy(portalLeft, portalApex); dtVcopy(portalRight, portalApex);可以看到这三个变量初始化为start点的值

-
我们从起始点(start)所在的多边形开始遍历,遍历次数(至少)为所有多边形的数量,在我们的例子中,数量为3;
for (int i = 0; i < pathSize; ++i) { -
在每次循环中,我们首先需要计算得到当前多边形与下一个多边形的公共边,得到公共边的两个点;假设当前多边形为绿色,下一个多边形为紫色,计算得到的两个点分别为
left、right,如下图

if (i+1 < pathSize) {unsigned char fromType; // fromType is ignored.// Next portal.if (dtStatusFailed(getPortalPoints(path[i], path[i+1], left, right, fromType, toType))){ -
然后我们使用三角形的有向面积(
signed area of triangle、叉乘)对几个点的位置进行判定/// Derives the signed xz-plane area of the triangle ABC, or the relationship of line AB to point C. /// @param[in] a Vertex A. [(x, y, z)] /// @param[in] b Vertex B. [(x, y, z)] /// @param[in] c Vertex C. [(x, y, z)] /// @return The signed xz-plane area of the triangle. inline float dtTriArea2D(const float* a, const float* b, const float* c) {const float abx = b[0] - a[0];const float abz = b[2] - a[2];const float acx = c[0] - a[0];const float acz = c[2] - a[2];return acx*abz - abx*acz; }首先是点
portalApex、portalRight、right,由于此时portalApex、portalRight相同,所以结果为0;

这个时候我们将点
portalRight置为点right;同理,对于点portalApex、portalLeft、left,将点portalLeft置为点left;

// Right vertex. if (dtTriArea2D(portalApex, portalRight, right) <= 0.0f) {if (dtVequal(portalApex, portalRight) || dtTriArea2D(portalApex, portalLeft, right) > 0.0f){dtVcopy(portalRight, right);rightPolyRef = (i+1 < pathSize) ? path[i+1] : 0;rightPolyType = toType;rightIndex = i;}else{ // ...... // Left vertex. if (dtTriArea2D(portalApex, portalLeft, left) >= 0.0f) {if (dtVequal(portalApex, portalLeft) || dtTriArea2D(portalApex, portalRight, left) < 0.0f){dtVcopy(portalLeft, left);leftPolyRef = (i+1 < pathSize) ? path[i+1] : 0;leftPolyType = toType;leftIndex = i;}else{至此,第一个循环结束
-
第二个循环,我们挪动第二个多边形,即当前多边形为第二个,计算第二、第三多边形的公共边,得到新的
left、right;如下图,当前多边形为绿色,下一个多边形为紫色,红色为公共边

-
同样,先对点
portalApex、portalRight、right以及点portalApex、portalLeft、left进行计算,这一步实际上是在判定边(left-right)是否在夹角(portalLeft-portalApex-portalRight)之内

首先是点right,由于点right在向量portalApex-portalRight左侧,不再进行下一步判定;即以下条件不满足:// Right vertex. if (dtTriArea2D(portalApex, portalRight, right) <= 0.0f) {接着是点
left,由于点left在向量portalApex-portalLeft左侧,满足条件,继续下一步判定;// Left vertex. if (dtTriArea2D(portalApex, portalLeft, left) >= 0.0f) {继续判定点
left是否在向量portalApex-portalRight左侧,结果满足条件,所以点portalRight是拐点if (dtVequal(portalApex, portalLeft) || dtTriArea2D(portalApex, portalRight, left) < 0.0f) {//...... } else {//....dtVcopy(portalApex, portalRight);得到拐点之后,将拐点置为新的
portalApex、portalRight、portalLeft,同时,下一次循环将继续从绿色多边形开始

此时,我们就得到了部分路径(上图中绿色点) -
继续下一次循环,同理可得到下述结果:

-
下一次循环,由于此时已经没有下一个多边形了,我们将
left、right置为点end

同样,先对点portalApex、portalRight、right以及点portalApex、portalLeft、left进行计算,

最终得到拐点portalRight

-
下一次循环,同样,由于此时已经没有下一个多边形了,我们将
left、right置为点end

继续对点portalApex、portalRight、right以及点portalApex、portalLeft、left进行计算,

此时循环结束,我们将点end加入路径}// Ignore status return value as we're just about to return anyway. appendVertex(closestEndPos, DT_STRAIGHTPATH_END, 0,straightPath, straightPathFlags, straightPathRefs,straightPathCount, maxStraightPath); -
最终结果,如下图绿色路径

-
寻路结果贴边优化的一种方式
- 通过上面的解释,我们知道寻路结果上的拐点都是多边形的边上的顶点,也就是
left、right点,所以想要让结果不那么贴边,可以将两点向内偏移,即:// Next portal. if (dtStatusFailed(getPortalPoints(path[i], path[i+1], left, right, fromType, toType))) {//...... }if (!dtVequal(left, right)) {float _left[3], _right[3];dtVcopy(_left, left);dtVcopy(_right, right);dtVlerp(left, _left, _right, 0.1);dtVlerp(right, _right, _left, 0.1); } - 优化后


相关文章:
【recast-navigation/源码解析】findStraightPath详解以及寻路结果贴边优化
说在前面 recast-navigation版本:1.6.0 叉积cross product 正常来讲,叉乘为: ∣ A ⃗ B ⃗ ∣ ∣ x A y A x B y B ∣ x A ⋅ y B − x B ⋅ y A |\vec{A} \times \vec{B}|\begin{vmatrix} x_A & y_A \\ x_B & y_B \end{vmatrix…...
移动管家手机智能控制汽车系统
手机可以通过下载特定的应用程序来控制汽车系统,实现远程启动、锁/解锁车门、调节车内温度等功能。 手机智能控制汽车系统主要通过下载并安装特定的APP来实现。 首先,用户需要确定自己的手机系统是安卓还是苹果版,然后前往应用…...
828华为云征文|华为云Flexus X实例Redis性能加速评测及对比
目录 前言 一、华为云Flexus X加速Redis购买 1.1 Flexus X实例购买 1.2 Redis加速镜像选择 1.3 重置密码 1.4 登录Flexus X实例 1.5 Flexus X实例Redis验证 二、Redis测评工具介绍 三、华为云Flexus X实例加速Redis测评 3.1 string类型 3.2 hash类型 3.3 list类型 3.4 set类型 …...
【OpenCV3】图像的翻转、图像的旋转、仿射变换之图像平移、仿射变换之获取变换矩阵、透视变换
1 图像的放大与缩小 2 图像的翻转 3 图像的旋转 4 仿射变换之图像平移 5 仿射变换之获取变换矩阵 6 透视变换 1 图像的放大与缩小 resize(src, dsize[, dst[, fx[, fy[, interpolation]]]]) src: 要缩放的图片dsize: 缩放之后的图片大小, 元组和列表表示均可.dst: 可选参数, 缩…...
不要认为996是开玩笑
996 预防针 随着秋招进程的不断推进,有部分同学已经 OC,有部分同学还在苦苦挣扎,并不断降低自己的预期,包括在和 HR 沟通过程中,主动说出自己愿意接受加班,愿意接受 996,以此来博得企业方面的加…...
精益工程师资格证书:2024年CLMP报名指南
随着全球对精益管理的需求日益增长,精益管理专业人士资格认证(CLMP)正成为越来越多精益工程师和精益管理人员提升职业竞争力的首选。作为一种注重管理而非生产的认证,CLMP不仅适用于制造业的专业人士,也吸引了各行业的…...
【Unity基础】如何选择脚本编译方式Mono和IL2CPP?
Edit -> Project Settings -> Player 在 Unity 中,Scripting Backend 决定了项目的脚本编译方式,即如何将 C# 代码转换为可执行代码。Unity 提供了两种主要的 Scripting Backend 选项:Mono 和 IL2CPP。它们之间的区别影响了项目的性能、…...
写在OceanBase开源三周年
我收获的深刻感触get 感触1:解决问题才有生存价值 [产品力] 感触2:永无止境的“易用性” [易用性] 感触3:立下“双赢”的flag 感触4:社区建设离不开用户和开发者参与 感触5:从易用到用户自助 [自助能力] 当时想法很简…...
【笔记】408刷题笔记
文章目录 三对角三叉树求最小带权路径UDP报文首部和TCP报文首部IP报文首部TCP报文首部UDP报文首部 刷新和再生的区别地址译码 为了区分队空队满,可以使用三种处理方式 1)牺牲一个单元 队头指针在队尾指针的下一位置作为队满的标志 队满条件:(…...
GitHub Star 数量前 13 的自托管项目清单
一个多月前,我们撰写并发布了这篇文章《终极自托管解决方案指南》。在那篇文章里我们深入探讨了云端服务与自托管方案的对比、自托管的潜在挑战、如何选择适合自托管解决方案,并深入介绍了五款涵盖不同场景的优秀自托管产品。 关于自托管的优势…...
js实现生成随机数值的数组
生成随机数值的数组 方法一:使用while循环和Set // min 开始数值, max 结束数值, count 数组内填充几个数值 function generateUniqueRandomNumbers(min, max, count) { let result new Set(); while (result.size < count) { let n…...
视频怎么转换成mp3格式?分享5种便捷的转换方法
在日常生活中,我们经常会遇到需要将视频文件中的音频提取出来,转换成MP3格式的情况,以便在手机、MP3播放器或其他设备上播放。今天,我将为大家介绍5种视频转MP3的方法,非常简单便捷,一起来学习下吧。 方法一…...
Reflection 70B如何革新语言模型的准确性与推理能力
在开源人工智能模型领域,HyperWrite 公司开发的 Reflection 70B 模型以其创新的“反射”机制成为新的重量级竞争者。这一模型旨在解决大型语言模型常见的“幻觉”问题,即生成不准确或虚构的信息。Reflection 70B 通过在提供最终响应之前评估和纠正自己的…...
覆盖索引是什么意思?
文章目录 Q1:覆盖索引是什么意思?覆盖索引的工作原理覆盖索引的优势覆盖索引的示例覆盖索引的使用场景覆盖索引的限制总结 Q2:为什么查询所涉及的所有字段都在索引中存在,那么数据库就无需回表?1. **索引本身存储了字段…...
最大间距问题
LeetCode164 最大间距 基数排序 #include <iostream> #include <vector> using namespace std;class Solution { public:int maximumGap(vector<int>& nums) {int nnums.size();if(n<2) return 0;int exp1;int Maxnums[0];vector<int> buf(n)…...
【Hadoop|MapReduce篇】Hadoop序列化概述
1. 什么是序列化 序列化就是把内存中的对象,转换成字节序列(或其他数据传输协议)以便于存储到磁盘(持久化)和网络传输。 反序列化就是将收到的字节序列(或其他数据传输协议)或者磁盘的持久化数…...
【Elasticsearch系列】Elasticsearch中的分页
💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…...
NLTK:一个强大的自然语言处理处理Python库
我是东哥,一名热爱技术的自媒体创作者。今天,我将为大家介绍一个非常有趣且强大的Python库——NLTK。无论你是刚刚接触Python的小白,还是对自然语言处理(NLP)有些许了解的朋友,NLTK都是一个值得学习的工具。…...
NUUO网络视频录像机 css_parser.php 任意文件读取漏洞复现
0x01 产品简介 NUUO网络视频录像机(Network Video Recorder,简称NVR)是NUUO Inc.生产的一种专业视频监控设备,它广泛应用于零售、交通、教育、政府和银行等多个领域。能够同时管理多个IP摄像头,实现视频录制、存储、回放及远程监控等功能。它采用先进的视频处理技术,提供…...
【支付】Stripe支付通道Java对接(产品 价格 支付 查询 退款 回调)
Stripe是一家美国科技公司,成立于2010年,由爱尔兰兄弟Patrick Collison和John Collison共同创立。该公司致力于提供高效、简洁的互联网支付收款服务,为开发者或商家提供支付API接口或代码,使商家的网站、移动APP支持信用卡付款。S…...
别再只用交叉熵了!医疗AI中疾病分级任务,试试PyTorch实现这个序数回归损失函数
医疗AI中的序数回归:超越交叉熵的疾病分级新范式 在医疗人工智能领域,我们经常遇到需要预测疾病严重程度分级的任务——从轻度到中度再到重度,这些类别之间存在明确的递进关系。传统做法是直接套用交叉熵损失函数,但这就像用尺子测…...
精准获取与高效转换:基于burst2safe的哨兵SLC burst数据轻量化处理实践
1. 哨兵SLC burst数据处理的必要性 处理卫星遥感数据时,我们常常面临一个两难选择:要么下载整景数据占用大量存储空间,要么难以精准获取研究区域的小范围数据。以Sentinel-1卫星为例,单景解压后的SLC数据可达7GB,而实际…...
别再为长文档发愁了!用DeepSeek-OCR + 单块A100,5步搞定古籍/财报批量识别
单块A100实战指南:用DeepSeek-OCR高效处理古籍与财报的5个关键步骤 当某省级图书馆需要数字化10万页明清古籍时,技术团队发现传统OCR方案需要3个月才能完成,而采用DeepSeek-OCR配合单块A100的方案,仅用11天就交付了准确率92%的数…...
动态规划专练:力扣第509、70、746题
由于对动态规划DP算法 掌握得不是很好,所以决定进行动态规划专项训练。动态规划五部曲①确定dp[i]含义②递推公式③dp数组如何初始化④遍历顺序⑤打印dp数组(debug)除了第五条在力扣上不开会员无法实现外,其余四项就是做出dp类型题…...
告别Keil?STM32CubeIDE环境搭建全记录:附JAVA安装与汉化资源指北
从Keil到STM32CubeIDE:嵌入式开发环境迁移实战指南 当ST官方逐渐将重心转向HAL库生态时,许多传统开发者正面临工具链升级的抉择。作为一款集成了STM32CubeMX功能的Eclipse-based IDE,STM32CubeIDE不仅代表着开发模式的转变,更预示…...
EB Tresos里XDM文件详解:不只是配置界面,更是你定制MCAL模块的‘源代码’
EB Tresos中XDM文件的深度解析:从配置界面到MCAL模块定制化开发 在AUTOSAR开发领域,EB Tresos Studio作为行业标准的MCAL配置工具,其核心机制往往隐藏在那些看似普通的配置文件中。XDM文件就是这样一个关键角色——它远不止是配置界面的数据源…...
STM32CubeMx 软件模拟SPI四种模式
(1)SPI的概念: SPI总线传输一共有4种模式,这4种模式分别由时钟极性(CPOL)和时钟相位(CPHA)来定义。 CPOL:规定了SCK时钟信号空闲状态的电平 CPHA:规定了数据是在SCK时钟的上升沿还是下降沿被采样 模式0&am…...
动态生成展示:LiuJuan20260223Zimage模型根据实时天气创作“风晴雨雪”主题画
动态生成展示:LiuJuan20260223Zimage模型根据实时天气创作“风晴雨雪”主题画 你有没有想过,家里的数字画框或者手机壁纸,能像有生命一样,随着窗外的天气实时变化?今天,我就带你体验一个特别有意思的玩法&…...
SUPER COLORIZER 数据库集成实践:MySQL管理海量图像处理任务与结果
SUPER COLORIZER 数据库集成实践:MySQL管理海量图像处理任务与结果 如果你正在管理一个需要批量处理成千上万张图片的项目,比如给老照片上色、统一调整产品图风格,或者为电商平台批量生成不同尺寸的图片,那你肯定遇到过这样的烦恼…...
飞书文档全流程备份终极方案:从手动操作到自动化管理的完美转型
飞书文档全流程备份终极方案:从手动操作到自动化管理的完美转型 【免费下载链接】feishu-doc-export 项目地址: https://gitcode.com/gh_mirrors/fe/feishu-doc-export 价值定位:破解企业文档管理的三大核心痛点 📊 在数字化办公日益…...
