服务器3D场景建模(七):四叉树的邻居关系

2023-11-09 03:11

本文主要是介绍服务器3D场景建模(七):四叉树的邻居关系,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

实测,经典四叉树效率比本中所述效率高!因此本文请无视之

四叉树的邻居节点?

常见的AOI使用Tile为基础,来实现。

每个Tile周围有8个邻居。因此在游戏对象移动或AOI时,可以O(1)的时间复杂度,定位8个邻居Tile。

而经典的四叉树代码实现,是没有邻居节点概念的。

图1,A节点的邻居节点:
图1

A节点有B、C、D、E、F、G邻居节点。

经典的四叉树代码实现,是需要从根节点开始遍历,才能够访问到邻居节点E、F、G。

图2,某AOI操作:

图2

红色区域的AOI,经典四叉树实现上,从根节点宽度优先遍历,按红色区域是否与节点区域有相交或包含作为条件,遍历之。时间复杂度为O(logN)

如果A节点知道自己的邻居节点,那么可以O(1)的时间复杂度,完成需要处理的节点定位。

有邻居关系的四叉树的AOI操作

图3,A上的红色区域的AOI:

图3

只要遍历A节点及其所有邻居节点,按红色区域是否与节点区域有相交或包含作为条件,遍历之;A的每个邻居节点递归重复操作。

以上操作与Tile上的AOI操作是同时间复杂度的。且比经典四叉树实现高效。

如何创建四叉树的邻居关系

图4,若N节点已经知道自己的邻居关系:

图4

图5,那么N节点分裂时,只要维护下孩子节点与邻居节点的关系即可:

图5

  1. L为N节点的邻居节点列表
  2. 遍历L,删除邻居节点对N的邻居信息
  3. N节点变成非叶节点,不再需要邻居信息,删除这些信息之
  4. N节点分裂为A、B、C、D4个孩子节点
  5. 对每个孩子节点,遍历L,根据节点区域是否相邻,构建孩子节点与L列表中节点的邻居信息

以上。

3种AOI对比

前提假设:游戏对象间有碰撞

算法占用内存AddLeaveMoveAOI说明
Tile算法 O(WH) O ( W ∗ H ) O(1) O ( 1 ) O(1) O ( 1 ) O(T) O ( T ) O(T) O ( T )
2维数组。
第1维表达所有Tile;
第2维表达每个Tile上的游戏对象列表
QuadTree算法 O(N) O ( N ) O(logN) O ( l o g N ) O(1) O ( 1 ) 通常 O(L) O ( L )
跨边界 O(logN)+O(L) O ( l o g N ) + O ( L )
O(logN)+O(L) O ( l o g N ) + O ( L ) 四叉树。
叶子节点上有游戏对象列表;
叶子节点游戏对象超过指定数量限制,则分裂子节点
叶子节点有游戏对象Leave,可能触发4兄弟节点合并
节点边界上的游戏对象放在父节点上
AOI从Root节点开始做广度遍历
QuadTree带邻居关系 O(N) O ( N ) O(logN) O ( l o g N ) O(1) O ( 1 ) O(L) O ( L ) O(L) O ( L ) 四叉树并带有邻居关系。
叶子节点上有游戏对象列表,邻居列表;
叶子节点游戏对象超过指定数量限制,则分裂子节点
叶子节点有游戏对象Leave,可能触发4兄弟节点合并
AOI从始发叶子节点开始,遍历邻居节点

上图,

  • N为游戏对象数量。
  • W为地图宽
  • H为地图长
  • T为Tile上能容纳的游戏对象数量
  • L为叶子节点区域能容纳的游戏对象数量

可以看出,理论上,地图越大,QuadTree优势越明显。

对于QuadTree带邻居关系,还忽略了一个重要问题,就是 非平衡数的邻居数量问题

非平衡数的邻居数量问题

比如下图中,在极端情况,F节点及其子节点持续分裂,一直分裂到4单位长宽大小的区域,才停止分裂(不好画自己想象)

图1

则E节点会有很多邻居。

如果是1024*1024的地图,则E节点会有128个邻居;如果是8192*8192的地图,则E节点会有1024个邻居。

因此,算法中,要对这种情况做特殊处理。

比如,如果E节点邻居数超过8个,则这里的AOI处理,使用经典四叉树算法。

这篇关于服务器3D场景建模(七):四叉树的邻居关系的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/373756

相关文章

使用Python实现快速搭建本地HTTP服务器

《使用Python实现快速搭建本地HTTP服务器》:本文主要介绍如何使用Python快速搭建本地HTTP服务器,轻松实现一键HTTP文件共享,同时结合二维码技术,让访问更简单,感兴趣的小伙伴可以了... 目录1. 概述2. 快速搭建 HTTP 文件共享服务2.1 核心思路2.2 代码实现2.3 代码解读3.

Tomcat版本与Java版本的关系及说明

《Tomcat版本与Java版本的关系及说明》:本文主要介绍Tomcat版本与Java版本的关系及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Tomcat版本与Java版本的关系Tomcat历史版本对应的Java版本Tomcat支持哪些版本的pythonJ

CentOS 7部署主域名服务器 DNS的方法

《CentOS7部署主域名服务器DNS的方法》文章详细介绍了在CentOS7上部署主域名服务器DNS的步骤,包括安装BIND服务、配置DNS服务、添加域名区域、创建区域文件、配置反向解析、检查配置... 目录1. 安装 BIND 服务和工具2.  配置 BIND 服务3 . 添加你的域名区域配置4.创建区域

Java中&和&&以及|和||的区别、应用场景和代码示例

《Java中&和&&以及|和||的区别、应用场景和代码示例》:本文主要介绍Java中的逻辑运算符&、&&、|和||的区别,包括它们在布尔和整数类型上的应用,文中通过代码介绍的非常详细,需要的朋友可... 目录前言1. & 和 &&代码示例2. | 和 ||代码示例3. 为什么要使用 & 和 | 而不是总是使

Windows Server服务器上配置FileZilla后,FTP连接不上?

《WindowsServer服务器上配置FileZilla后,FTP连接不上?》WindowsServer服务器上配置FileZilla后,FTP连接错误和操作超时的问题,应该如何解决?首先,通过... 目录在Windohttp://www.chinasem.cnws防火墙开启的情况下,遇到的错误如下:无法与

Java中Runnable和Callable的区别和联系及使用场景

《Java中Runnable和Callable的区别和联系及使用场景》Java多线程有两个重要的接口,Runnable和Callable,分别提供一个run方法和call方法,二者是有较大差异的,本文... 目录一、Runnable使用场景二、Callable的使用场景三、关于Future和FutureTa

Windows server服务器使用blat命令行发送邮件

《Windowsserver服务器使用blat命令行发送邮件》在linux平台的命令行下可以使用mail命令来发送邮件,windows平台没有内置的命令,但可以使用开源的blat,其官方主页为ht... 目录下载blatBAT命令行示例备注总结在linux平台的命令行下可以使用mail命令来发送邮件,Win

Ubuntu 22.04 服务器安装部署(nginx+postgresql)

《Ubuntu22.04服务器安装部署(nginx+postgresql)》Ubuntu22.04LTS是迄今为止最好的Ubuntu版本之一,很多linux的应用服务器都是选择的这个版本... 目录是什么让 Ubuntu 22.04 LTS 变得安全?更新了安全包linux 内核改进一、部署环境二、安装系统

nginx配置多域名共用服务器80端口

《nginx配置多域名共用服务器80端口》本文主要介绍了配置Nginx.conf文件,使得同一台服务器上的服务程序能够根据域名分发到相应的端口进行处理,从而实现用户通过abc.com或xyz.com直... 多个域名,比如两个域名,这两个域名其实共用一台服务器(意味着域名解析到同一个IP),一个域名为abc

pycharm远程连接服务器运行pytorch的过程详解

《pycharm远程连接服务器运行pytorch的过程详解》:本文主要介绍在Linux环境下使用Anaconda管理不同版本的Python环境,并通过PyCharm远程连接服务器来运行PyTorc... 目录linux部署pytorch背景介绍Anaconda安装Linux安装pytorch虚拟环境安装cu