UE NavigationSystem的相关实现
导航数据的构建流程
导航数据的收集
导航系统中绑定了Actor、Component注册完成以及取消时的委托,通过这些委托把数据及时更新到导航系统的八叉树结构中
导航系统的辅助结构DefaultOctreeController、DefaultDirtyAreasController分别承担了空间数据查询和置脏区域重新构建的任务,后者会把置脏数据最终投递到导航数据生成器的待处理结构里最终执行生成操作
开启构建操作(静态构建)
开启导航数据构建有几种情况:开启了动态导航、指令集开启、编辑器模式下的自动构建开启、调整导航偏移导航失效造成的重新构建。构建执行流程大致如下:
当开启导航构建后导航系统首先重置掉待处理区域变更请求以及控制置脏数据的管理器,然后迭代所有的ARecastNavMesh执行同名构建函数RebuildAll
ARecastNavMesh首先会检查是否有正在构建操作如果存在操作则取消并重置,构建新的导航数据生成器实例并执行初始化,在初始化中进行有效区域边界更新并置脏所有的边界
把边界数据投递到待处理结构中,然后通过执行构建生成器的EnsureBuildCompletion函数确认所有的构建工作完成,实质上是把所有的待处理结构打包成异步或者同步执行的任务来处理
其中值得提及的是收集碰撞数据的方法:导航系统的辅助结构DefaultOctreeController中的NavOctree与DirtyBound做交叉检测,符合的则做数据收集
使用瓦片生成器结构构建瓦片数据
瓦片导航数据生成完成以后会把生成的数据添加到ARecastNavMesh数据真正的持有者DetourNavMesh(dtNavMesh)中
动态导航数据与静态生成区别:动态导航模式下导航系统中挂载的Actor、Component注册取消委托受场景变更的影响会不断的更新,进而推动整个构建导航逻辑的执行(而且在动态模式下这些具有碰撞数据的实例会及时导出碰撞数据)。
生成瓦片数据的逻辑
当FRecastTileGeneratorTask异步执行或者同步执行一个构建的任务时FRecastTileGenerator::DoWork的调用就开始了,FRecastTileGenerator结构封装了生成单个瓦片的功能逻辑(封装Recast导航数据生成流程),DoWork的执行流程如下:
完成几何模型碰撞数据收集和缓存,其中包括顶点数据、索引数据、可行走斜坡结构数据、修改器、导航连接、瓦片原生几何数据等
构建高度场域数据(rcHeightfield)
计算光栅化遮罩(对导航Modifier进行处理)
光栅化或者说体素化三角形面(其中包括几何Transform数据从UE坐标到Recast的坐标转换)
标记可行走区域
体素边界过滤,防止边界外生成体素数据,通过InclusionBounds边界框判定(ApplyVoxelFilter)
Recast的生成过滤(WalkableClimb、walkableHeight)
生成紧缩高度场域(CompactHeightField)
根据walkableRadius剔除边缘(rcErodeWalkableAndLowAreas)
根据选择的算法生成区域数据(rcBuildDistanceField)
生成导航数据保存到瓦片生成器中
回收脚手架结构内存
导航数据的查询
导航系统封装的查询接口
K2_ProjectPointToNavigation 根据传入的点获取在NavMesh上的投影位置
K2_GetRandomReachablePointInRadius 获取指定半径内随机的一个导航可达点位
K2_GetRandomLocationInNavigableRadius 在给定的原点半径内的导航空间中生成一个随机位置
GetPathCost 获取路径消耗,不推荐使用
GetPathLength 获取路径长度, 不推荐使用
FindPathToLocationSynchronously 同步地获取两点间的导航路径
FindPathToActorSynchronously 同步地获取到达指定Actor的导航路径
以K2_GetRandomReachablePointInRadius为例说明导航数据查询路径
调用核心ARecastNavMesh的GetRandomPointInNavigableRadius函数
ARecastNavMesh很多时候是空壳逻辑它会把逻辑转交给FPImplRecastNavMesh
FPImplRecastNavMesh结构的ProjectPointToNavMesh函数会构建出一个dtNavMeshQuery查询结构来完成Ploy2D的相关查询
导航系统的配置参数说明
属性名字 | 属性描述 |
DefaultAgentName | 默认的导航代理名字,用于适配、查找默认的导航配置、导航数据,为空的情况下默认会把代理配置中的第一项作为默认配置 |
CrowdManagerClass | 定义导航系统使用的自定义CrowdManager基类,目前不需要修改 |
bAutoCreateNavigationData | 是否自动创建导航数据,通常情况下会在地图加载完成时、导航体积发生变更执行操作 |
bSpawnNavDataInNavBoundsLevel | 是否允许在子关卡中生成导航数据,否则在持久关卡中创建 |
bAllowClientSideNavigation | 是否允许客户端开启导航,默认客户端导航模式是关闭的,关闭的情况下整个导航系统不会创建 |
bShouldDiscardSubLevelNavData | 是否要丢弃子关卡中的导航数据,该操作在地图加载完成后通过委托回调进行检测完成。如果是勾选状态则整个子关卡中的RecastNavMesh会被销毁占用的内存会被释放 |
bTickWhilePaused | 当世界关卡处于Paused状态时是否要停止导航系统的Tick |
bInitialBuildingLocked | 在导航系统初始化完成或者导航系统重新加载完成的时候是否锁定导航系统的构建,如果为true只有在释放锁定标记后才能进行导航数据构建 |
bSkipAgentHeightCheckWhenPickingNavData | 在通过NavConfigProps进行导航数据匹配时,是否忽略对代理高度的检查 |
DataGatheringMode | 影响导航数据收集,注释说设置在构建碰撞信息时应如何收集导航数据。它的实质是在世界场景下Actor中的组件注册完成后导航系统中的OnComponentRegistered会被调用进而执行RegisterComponentToNavOctree,最终会执行到AddElementToNavOctree该函数会调用FNavigationOctree::AddNode函数,这个函数是最终的调用,它会执行ComponentExportDelegate委托的绑定,而其中绑定的是几何导出的逻辑处理FRecastNavMeshGenerator::ExportComponentGeometry |
GeometryExportVertexCountWarningThreshold | 当正在执行导出的数据结构中,定点数据容量大于此值时会产生日记记录,可参见ValidateGeometryExport |
DirtyAreaWarningSizeThreshold | 当该值为正值,向置脏控制器中添加的置脏区域(2D)大于该值时,产生警告日志 |
GatheringNavModifiersTimeLimitWarning | 如果 GatheringNavModifiersWarningLimitTime 为正数,如果调用 GetNavigationData 所花费的时间超过 GatheringNavModifiersWarningLimitTime,它将打印警告 |
bGenerateNavigationOnlyAroundNavigationInvokers | 是否仅在指定的目标对象周围生成导航数据,一旦开启全图可导航,依赖于导航动态生成,依赖于NavigationInvokerComponent |
ActiveTilesUpdateInterval | 激活的Tiles更新间隔依赖于bGenerateNavigationOnlyAroundNavigationInvokers |
SupportedAgents | 支持的代理导航配置 |
SupportedAgentsMask | 导航系统对支持的代理控制 |
导航数据查询规则
通过导航配置FNavAgentProperties结构查询导航数据实例时会发现该结构的IsEquivalent函数起着至关重要的作用,而该函数中起决定因素的值是AgentRadius、AgentHeight,当这两个参数各自的差值小于默认精度5.f则认为这两个导航代理配置是等价的,因此有以下匹配规则:
AgentToNavDataMap中通过FNavAgentProperties类型的Key查询到指定的NavData返回
如果无法查询到最佳匹配项则按照最优匹配原则获取指定导航数据配置查询AgentToNavDataMap返回
所有NavData实例都会在AgentToNavDataMap、NavDataSet中注册
WP情况下对导航的影响
WP下多了对区域导航数据的流入流出支持,它的核心实质是根据导航数据构建支持流入的Actor实例ANavigationDataChunkActor,该实例中保存了生成的导航数据URecastNavMeshDataChunk(TArray Tiles)。当该Actor实例被装载时携带的导航数据会替换掉DetourNavMesh中的Tile数据,卸载的时候又从DetourNavMesh移除。
Navigation Mesh参数说明
属性名字 | 属性描述 |
CellSize | 水平方向上体素尺寸tiles大小应设置为每侧32到128个cells之间。这将在运行时重建tiles时提供最佳性能 |
CellHeight | 垂直方向上体素尺寸 |
AgentRadius | 代理半径 |
AgentHeight | 代理高度 |
AgentMaxSlope | 代理可以移动的最大斜率(角度) |
AgentMaxStepHeight | Agent的可攀爬高度 |
MinRegionArea | 允许形成孤岛区域的最小单元数 |
MergeRegionSize | 分水岭算法中跨度计数小于此值的任何区域都将与更大的区域合并 |
MaxSimplificationError | 细节网格表面偏离高度场的最大距离 |
TileSizeUU | 单个tile的大小 |
TilePoolSize | NavMesh可以容纳的最大图块数 |
bFixedTilePoolSize | 如果为真,NavMesh将为Tile分配固定大小的容量,开启以支持流式传输 |
MaxSimultaneousTileGenerationJobsCount | 设置一次运行的异步瓦片生成器的数量限制,也用于一些同步任务 |
TileNumberHardLimit | 导航网格图块数量的绝对硬性限制。在使用带有navmesh的大地图进行修改时要非常非常小心。一个空的tile需要176个字节,并且空的tile是预先分配的 |
DefaultMaxSearchNodes | 指定执行导航查询时使用的A*节点的默认上限限制 |
DefaultMaxHierarchicalSearchNodes | 指定执行分层导航查询时使用的A*节点的默认上限限制 |
RegionChunkSplits | 用于区域分区的块分割数(沿单轴),仅用于ChunkyMonotone算法 |
LayerChunkSplits | 用于区域分区的层分割数(沿单轴),仅用于ChunkyMonotone算法 |
bSortNavigationAreasByCost | 控制导航区域在导航网格生成期间应用到导航网格之前是否按成本排序。 当区域重叠时,这是相关的,我们也希望区域成本表达区域相关性。 将其设置为 true 将导致区域按成本排序,但它也会增加导航网格生成成本一点 |
bIsWorldPartitioned | 是否是world partitioned map |
bPerformVoxelFiltering | 是否执行体素过滤 |
bMarkLowHeightAreas | 标记上方自由高度不足的区域,而不是将其切掉(仅适用于使用替换模式的区域修改器) |
bUseExtraTopCellWhenMarkingAreas | 应用于导航网格时,将区域导航修改器的边界顶部扩展一个单元格高度 |
bFilterLowSpanSequences | 如果跨度上方的间隙小于指定高度,则将可步行跨度标记为不可步行。 |
bFilterLowSpanFromTileCache | 如果设置,只有具有相应区域修改器的低高度跨度将存储在图块缓存中(减少内存,如果没有完整的图块重建就无法修改) |
bDoFullyAsyncNavDataGathering | navmesh 数据收集永远不会发生在游戏线程上,只会在后台线程上完成 |
导航数据的内存占用
NavOctree memory :导航八叉树的空间占用
ARecastNavMesh memory : DetourNavMesh中的所有瓦片结构内存总和 + 激活路径分配的内存(ActivePaths)+ 支持的区域结构分配的内存(SupportedAreas)+ 查询过滤器分配的内存(QueryFilters)+ 区域映射Map分配的内存(AreaClassToIdMap)+ 导航生成器内存占用(待处理任务数据结构大小 + 生成器结构本身大小 + 运行时任务结构大小 + 瓦片构建器分配的内存总和)
导航数据生成相关验证
多代理下的导航数据验证
支持的代理越多产生的数据容量越大,而且不同的代理配置生成的数据容量也是不同的
相关文章:
UE NavigationSystem的相关实现
导航数据的构建流程导航数据的收集导航系统中绑定了Actor、Component注册完成以及取消时的委托,通过这些委托把数据及时更新到导航系统的八叉树结构中导航系统的辅助结构DefaultOctreeController、DefaultDirtyAreasController分别承担了空间数据查询和置脏区域重新…...
Java 继承
文章目录1. 继承概述2. 变量的访问特点3. super 关键字4. 构造方法的访问特点5. 成员方法的访问特点6. 方法重写7. 继承案例1. 继承概述 继承是面向对象三大特征之一。可以使得子类具有父类的属性和方法,还可以在子类中重新定义,追加属性和方法。 publ…...
Python学习笔记8:异常
异常 一些内置的异常类 类名描述Exception几乎所有的异常类都是从它派生而来的AttributeError引用属性或给它赋值失败时引发OSError操作系统不能执行指定的任务(如打开文件)时引发,有多个子类IndexError使用序列中不存在的索引时引发&#…...
python保留小数函数总结
python保留小数——‘%f’‘%.nf’% x(定义的变量) 例子:a 82.16332 print(%.1f% a) print(%.2f% a) print(%.3f% a) print(%.4f% a) print(%.10f% a)输出结果python保留小数——format()函数Python2.6 开始ÿ…...
狐狸优化算法(Matlab代码实现)
👨🎓个人主页:研学社的博客💥💥💞💞欢迎来到本博客❤️❤️💥💥🏆博主优势:🌞🌞🌞博客内容尽量做到思维缜密…...
浏览器自动化框架沦为攻击者的工具
5月27日消息,安全公司Team Cymru的研究人员表示,越来越多的威胁参与者正在使用免费的浏览器自动化框架作为其攻击活动的一部分。 研究人员表示,该框架的技术准入门槛故意保持在较低水平,以创建一个由内容开发者和贡献者组成的活跃…...
SQL必备知识(自用)
数据库基础知识sql和mysql的区别:数据库查询大全(select)1、select 字段名 from 表;2、In查询:用于过滤你所需要查询的内容3、范围查询:between4、模糊查询:like5、查询空值/非空:is…...
BI工具术语表大全:从字母A-Z全面收录
谈到商业智能行业,变革是不可避免的。为了跟上步伐,各种各样的BI 解决方案正在快速迭代更新,以满足企业的数字化需求,那么市场上BI 工具种类繁杂,到底如何选择适合功能全面、满足自己企业运转情况的、合适的BI 工具呢&…...
vue3 + vite + ts 集成mars3d
vue3 vite ts 集成mars3d 文章目录vue3 vite ts 集成mars3d前言一、创建一个vue3 vite ts项目二、引入mars3d相关依赖三、vite.config.ts 相关配置四、 新建DIV容器 创建地图前言 使用mars3d过程中,需要集成mars3d到自己的项目中,mars3d开发教程…...
跨境卖家必看的沃尔玛Walmart商家入驻教程
沃尔玛Walmart作为作为全球连锁超市的鼻祖,有不可比拟的知名度。当沃尔玛从线下延伸到线上后,就成为一个自带IP与流量的线上平台,在全世界都拥有数量庞大的消费者群体。所以龙哥就结合自己注册Walmart的过程给大家详细讲解一下。 Walmart卖家…...
【GANs】什么是饱和损失函数 Non-Saturating LossFunction
Saturating VS Non-Saturating Loss functions in GANs【GANs】什么是饱和损失函数 Non-Saturating LossFunctionSaturating VS Non-Saturating Loss functions in GANs 饱和Loss 普通GAN loss是生成器希望最小化被判断为假的概率。x取值范围是[0,1],所以图中函数…...
USB接口虚拟网卡
1 基本概念 1.1 USB转以太网 - ASIX 4-byte length header before every ethernet packet. - Microchip LAN7800 128x32 bit Descriptor RAM, 32 bits DP_DATA address offset 030h for Descriptor RAM access. - Windows CMD参数格式: route /? -> Linux -h …...
基于SpringBoot的外卖项目的优化
基于SpringBoot的外卖项目的优化1、缓存优化1.1、缓存短信验证码问题分析代码改造1.2、缓存菜品数据实现思路1.3、Spring Cache介绍常用注解CachePutCacheEvictCacheable使用方式1.4、缓存套餐数据实现思路代码改造2、读写分离2.1、主从复制存在的问题介绍配置配置主库--master…...
Ubuntu20.04/22.04 ESP32 命令行开发环境配置
ESP32 芯片系列 ESP32分三个系列 ESP32-S ESP32-S3: Xtensa 32位 LX7 双核 240 MHz, 384KB ROM, 512KB SRAM, QFN7x7, 56-pin, 2.4G Wi-Fi BTESP32-S2: Xtensa 32位 LX7 单核 240 MHz, 128KB ROM, 320KB SRAM, QFN7x7, 56-pin, 2.4G Wi-Fi ESP32-C ESP32-C3: RISC-V 32位 单…...
Kali Linux使用Metasploit生成木马入侵安卓系统
额,这是我最后一篇文章了,周一我们开学了 文章目录前言一、Metasploit是什么?演示环境二、生成可执行木马文件1.生成2.运行命令并生成木马配置参数入侵安卓手机命令1.查看对方手机系统信息查看对方手机安装哪些app文件总结前言 前言…...
数据库复习1
一. 简答题(共1题,100分) 1. (简答题) 存在数据库test,数据库中有如下表: 1.学生表 Student(Sno,Sname,Sage,Ssex) --Sno 学号,Sname 学生姓名,Sage 出生年月,Ssex 学生性别 主键Sno 2.教师表 Teacher(Tno,Tname) --T…...
18. linux系统基础
shell 命令解析器 命令解析器作用: 他把在终端上输出的命令 给你解析成内核可以识别的指令,内核 是经过命令解析器的加工 shell在找命令的时候,所包含的路径,就是在这些路径里去 找 找到就执行 找不到就报错 报错 要么 这个命…...
ssh远程登录报错:kex_exchange_identification: Connection closed by remote host
基本信息系统:MacOS Catalina 10.15.7报错信息:终端登录远程服务器时报错:kex_exchange_identification: Connection closed by remote host复制然而服务商的一键登录或VNC登录正常。解决方案首先使用以下命令debug登录过程,以便定…...
Quartus II 的入门级使用
好久没有用VHDL写东西了,今天需要完成一个项目,重新复习一下新建工程新建工程file-->New Project Wizard, next, 选择存放的路径名字(projecttop-level 名字要相同),next,File name名字同上,…...
Java EE|TCP/IP协议栈之TCP协议工作机制上
文章目录前言一、确认应答二、超时重传三、连接管理三次握手四次挥手前言 前边,我们已经大概交代了TCP的报文结构。但是仍有一些字段我们不确定到底怎么理解,这里就分析TCP的内部工作机制了。 TCP的内部很复杂,有很多机制,这里我们…...
React 第五十五节 Router 中 useAsyncError的使用详解
前言 useAsyncError 是 React Router v6.4 引入的一个钩子,用于处理异步操作(如数据加载)中的错误。下面我将详细解释其用途并提供代码示例。 一、useAsyncError 用途 处理异步错误:捕获在 loader 或 action 中发生的异步错误替…...
K8S认证|CKS题库+答案| 11. AppArmor
目录 11. AppArmor 免费获取并激活 CKA_v1.31_模拟系统 题目 开始操作: 1)、切换集群 2)、切换节点 3)、切换到 apparmor 的目录 4)、执行 apparmor 策略模块 5)、修改 pod 文件 6)、…...
第25节 Node.js 断言测试
Node.js的assert模块主要用于编写程序的单元测试时使用,通过断言可以提早发现和排查出错误。 稳定性: 5 - 锁定 这个模块可用于应用的单元测试,通过 require(assert) 可以使用这个模块。 assert.fail(actual, expected, message, operator) 使用参数…...
css的定位(position)详解:相对定位 绝对定位 固定定位
在 CSS 中,元素的定位通过 position 属性控制,共有 5 种定位模式:static(静态定位)、relative(相对定位)、absolute(绝对定位)、fixed(固定定位)和…...
PL0语法,分析器实现!
简介 PL/0 是一种简单的编程语言,通常用于教学编译原理。它的语法结构清晰,功能包括常量定义、变量声明、过程(子程序)定义以及基本的控制结构(如条件语句和循环语句)。 PL/0 语法规范 PL/0 是一种教学用的小型编程语言,由 Niklaus Wirth 设计,用于展示编译原理的核…...
工业自动化时代的精准装配革新:迁移科技3D视觉系统如何重塑机器人定位装配
AI3D视觉的工业赋能者 迁移科技成立于2017年,作为行业领先的3D工业相机及视觉系统供应商,累计完成数亿元融资。其核心技术覆盖硬件设计、算法优化及软件集成,通过稳定、易用、高回报的AI3D视觉系统,为汽车、新能源、金属制造等行…...
微信小程序云开发平台MySQL的连接方式
注:微信小程序云开发平台指的是腾讯云开发 先给结论:微信小程序云开发平台的MySQL,无法通过获取数据库连接信息的方式进行连接,连接只能通过云开发的SDK连接,具体要参考官方文档: 为什么? 因为…...
【学习笔记】深入理解Java虚拟机学习笔记——第4章 虚拟机性能监控,故障处理工具
第2章 虚拟机性能监控,故障处理工具 4.1 概述 略 4.2 基础故障处理工具 4.2.1 jps:虚拟机进程状况工具 命令:jps [options] [hostid] 功能:本地虚拟机进程显示进程ID(与ps相同),可同时显示主类&#x…...
Android 之 kotlin 语言学习笔记三(Kotlin-Java 互操作)
参考官方文档:https://developer.android.google.cn/kotlin/interop?hlzh-cn 一、Java(供 Kotlin 使用) 1、不得使用硬关键字 不要使用 Kotlin 的任何硬关键字作为方法的名称 或字段。允许使用 Kotlin 的软关键字、修饰符关键字和特殊标识…...
Java多线程实现之Thread类深度解析
Java多线程实现之Thread类深度解析 一、多线程基础概念1.1 什么是线程1.2 多线程的优势1.3 Java多线程模型 二、Thread类的基本结构与构造函数2.1 Thread类的继承关系2.2 构造函数 三、创建和启动线程3.1 继承Thread类创建线程3.2 实现Runnable接口创建线程 四、Thread类的核心…...
