C++自研游戏引擎-碰撞检测组件-八叉树AABB检测算法实现

八叉树碰撞检测是一种在三维空间中高效处理物体碰撞检测的算法,其原理可以类比为一个管理三维空间物体的智能系统。这个示例包含两个部分:八叉树部分用于宏观检测,AABB用于微观检测。AABB可以更换为均值或节点检测来提高检测精度。

八叉树的构建

  1. 确定根节点范围
    首先要为整个碰撞检测系统确定一个初始范围,这就像是为所有参与碰撞检测的物体划定一个 “活动区域”。这个范围是一个能够完全容纳所有待检测物体的三维立方体空间,它构成了八叉树的根节点。
  2. 递归分割空间
    为了更高效地管理和查找物体,八叉树会对这个初始的大立方体空间进行递归分割。具体做法是沿着三个坐标轴的中点,将大立方体分割成八个小立方体,每个小立方体对应根节点的一个子节点。之后,系统会检查每个子节点所包含的物体数量:
    若某个子节点中的物体数量小于预设的阈值,就认为该区域内的物体分布较为稀疏,无需再进行分割,这些物体就存储在该节点中。
    若物体数量超过阈值,说明该区域物体较为密集,需要进一步细分。于是会将这个节点的空间继续分割成八个更小的子空间,并对每个子空间重复上述检查过程,直到满足停止条件。
    碰撞检测过程
  3. 插入物体
    在将物体的轴对齐包围盒(AABB)插入八叉树时,系统会从根节点开始判断物体的 AABB 与当前节点的空间是否相交:
    如果不相交,表明该物体不在当前节点所管理的空间范围内,无需在此节点存储该物体。
    如果相交,则将物体插入当前节点。若当前节点已经被分割成子节点,系统会进一步判断物体的 AABB 与哪个子节点的空间相交,并将物体插入对应的子节点中。
  4. 查询碰撞
    当需要检测某个物体(用其 AABB 表示)是否与其他物体发生碰撞时,系统会从八叉树的根节点开始查询:
    若该物体的 AABB 与当前节点的空间不相交,说明该节点及其子节点中的物体都不可能与该物体发生碰撞,无需继续检查该节点及其子树。
    若相交,则检查当前节点中存储的物体的 AABB 与该物体的 AABB 是否相交。
    若当前节点有子节点,系统会递归地对每个子节点进行相同的查询操作,直到遍历完所有可能发生碰撞的节点。
    八叉树碰撞检测的优缺点
    优点
    高效性:通过对三维空间进行递归分割,八叉树将碰撞检测的范围缩小到可能发生碰撞的区域,避免了对所有物体进行两两比较,从而显著减少了不必要的计算,提高了碰撞检测的效率。在处理大量物体的场景中,这种优势更为明显。
    适应性:八叉树能够根据物体在空间中的实际分布情况自适应地进行空间划分,对于物体分布不均匀的场景也能有效地组织和管理物体。
    缺点
    构建和维护成本较高:构建八叉树需要对空间进行递归分割,并将物体分配到相应的节点中,这需要一定的时间和空间开销。特别是在物体频繁移动或新增、删除物体的场景中,需要不断更新八叉树的结构,增加了维护成本。
    存在精度问题:使用 AABB 来近似表示物体可能会导致一定的精度损失,尤其是对于形状复杂的物体,AABB 可能无法精确地描述其外形,从而产生误判。

C++代码

#include <iostream>
#include <vector>
#include <memory>

// 定义三维向量结构体
struct Vec3 {
    float x, y, z;
    Vec3(float x = 0, float y = 0, float z = 0) : x(x), y(y), z(z) {}
};

// 定义 AABB 结构体
struct AABB {
    Vec3 min;
    Vec3 max;

    AABB(const Vec3& min, const Vec3& max) : min(min), max(max) {}

    // 判断两个 AABB 是否相交
    bool intersects(const AABB& other) const {
        return (min.x <= other.max.x && max.x >= other.min.x) &&
               (min.y <= other.max.y && max.y >= other.min.y) &&
               (min.z <= other.max.z && max.z >= other.min.z);
    }
};

// 定义八叉树节点类
class OctreeNode {
public:
    AABB bounds;
    std::vector<AABB> objects;
    std::vector<std::unique_ptr<OctreeNode>> children;

    OctreeNode(const AABB& bounds) : bounds(bounds) {}

    // 插入 AABB 到节点中
    void insert(const AABB& object) {
        if (children.empty()) {
            if (objects.size() < 8) {
                objects.push_back(object);
            } else {
                split();
                insert(object);
            }
        } else {
            for (auto& child : children) {
                if (child->bounds.intersects(object)) {
                    child->insert(object);
                }
            }
        }
    }

    // 分割节点
    void split() {
        Vec3 center((bounds.min.x + bounds.max.x) / 2, (bounds.min.y + bounds.max.y) / 2, (bounds.min.z + bounds.max.z) / 2);

        children.resize(8);
        children[0] = std::make_unique<OctreeNode>(AABB(bounds.min, center));
        children[1] = std::make_unique<OctreeNode>(AABB(Vec3(center.x, bounds.min.y, bounds.min.z), Vec3(bounds.max.x, center.y, center.z)));
        children[2] = std::make_unique<OctreeNode>(AABB(Vec3(bounds.min.x, center.y, bounds.min.z), Vec3(center.x, bounds.max.y, center.z)));
        children[3] = std::make_unique<OctreeNode>(AABB(Vec3(center.x, center.y, bounds.min.z), Vec3(bounds.max.x, bounds.max.y, center.z)));
        children[4] = std::make_unique<OctreeNode>(AABB(Vec3(bounds.min.x, bounds.min.y, center.z), Vec3(center.x, center.y, bounds.max.z)));
        children[5] = std::make_unique<OctreeNode>(AABB(Vec3(center.x, bounds.min.y, center.z), Vec3(bounds.max.x, center.y, bounds.max.z)));
        children[6] = std::make_unique<OctreeNode>(AABB(Vec3(bounds.min.x, center.y, center.z), Vec3(center.x, bounds.max.y, bounds.max.z)));
        children[7] = std::make_unique<OctreeNode>(AABB(center, bounds.max));

        for (const auto& object : objects) {
            for (auto& child : children) {
                if (child->bounds.intersects(object)) {
                    child->insert(object);
                }
            }
        }
        objects.clear();
    }

    // 检测与指定 AABB 的碰撞
    void query(const AABB& object, std::vector<AABB>& result) const {
        if (bounds.intersects(object)) {
            for (const auto& obj : objects) {
                if (obj.intersects(object)) {
                    result.push_back(obj);
                }
            }
            for (const auto& child : children) {
                child->query(object, result);
            }
        }
    }
};

// 定义八叉树类
class Octree {
public:
    std::unique_ptr<OctreeNode> root;

    Octree(const AABB& bounds) : root(std::make_unique<OctreeNode>(bounds)) {}

    // 插入 AABB 到八叉树中
    void insert(const AABB& object) {
        root->insert(object);
    }

    // 检测与指定 AABB 的碰撞
    std::vector<AABB> query(const AABB& object) const {
        std::vector<AABB> result;
        root->query(object, result);
        return result;
    }
};

// 示例使用
int main() {
    // 定义八叉树的边界
    AABB octreeBounds(Vec3(0, 0, 0), Vec3(100, 100, 100));
    Octree octree(octreeBounds);

    // 插入一些 AABB
    octree.insert(AABB(Vec3(10, 10, 10), Vec3(20, 20, 20)));
    octree.insert(AABB(Vec3(30, 30, 30), Vec3(40, 40, 40)));

    // 定义一个查询的 AABB
    AABB queryAABB(Vec3(15, 15, 15), Vec3(25, 25, 25));

    // 进行碰撞检测
    std::vector<AABB> collisions = octree.query(queryAABB);

    // 输出碰撞结果
    std::cout << "Collisions found: " << collisions.size() << std::endl;
    for (const auto& collision : collisions) {
        std::cout << "Collision: min(" << collision.min.x << ", " << collision.min.y << ", " << collision.min.z << "), max("
                  << collision.max.x << ", " << collision.max.y << ", " << collision.max.z << ")" << std::endl;
    }

    return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

1、Prometheus 监控系统(上)

Prometheus 监控系统&#xff08;上&#xff09; 认识一下 PrometheusPrometheus 的特点Prometheus 的生态组件Prometheus 的工作模式Prometheus 的工作流程Prometheus 的局限性&#xff1a; 部署 PrometheusPrometheust Server 端安装和相关配置部署 Exporters部署 Node Expor…

【设计模式】-工厂模式(简单工厂、工厂方法、抽象工厂)

工厂模式(简单工厂、工厂方法、抽象工厂) 介绍 简单工厂模式 简单工厂模式不属于23种GoF设计模式之一&#xff0c;但它是一种常见的设计模式。它提供了一种创建对象的接口&#xff0c;但由子类决定要实例化的类是哪一个。这样&#xff0c;工厂方法模式让类的实例化推迟到子类…

应急响应(linux 篇,以centos 7为例)

一、基础命令 1.查看已经登录的用户w 2.查看所有用户最近一次登录&#xff1a;lastlog 3.查看历史上登录的用户还有登录失败的用户 历史上所有登录成功的记录 last /var/log/wtmp 历史上所有登录失败的记录 Lastb /var/log/btmp 4.SSH登录日志 查看所有日志&#xff1a;…

【实测】用全志A733平板搭建一个端侧Deepseek算力平台

随着DeepSeek 的蒸馏技术的横空出世&#xff0c;端侧 SoC 芯片上运行大模型成为可能。那么端侧芯片跑大模型的效果如何呢&#xff1f;本文将在全志 A733 芯片平台上部署一个 DeepSeek-R1:1.5B 模型&#xff0c;并进行实测效果展示。 端侧平台环境 设备&#xff1a;全志A733平板…

nuxt中引入element-ui组件控制台报错问题

在使用element-ui组件的外层加一层 <client-only placeholder"Loading..."><van-button type"primary">主要按钮</van-button> </client-only> 实际使用&#xff1a; <div class"tab"><client-only placehol…

数据结构(考研)

线性表 顺序表 顺序表的静态分配 //线性表的元素类型为 ElemType//顺序表的静态分配 #define MaxSize10 typedef int ElemType; typedef struct{ElemType data[MaxSize];int length; }SqList;顺序表的动态分配 //顺序表的动态分配 #define InitSize 10 typedef struct{El…

【广州大学主办,发表有保障 | IEEE出版,稳定EI检索,往届见刊后快至1个月检索】第二届电气技术与自动化工程国际学术会议 (ETAE 2025)

第二届电气技术与自动化工程国际学术会议 (ETAE 2025) The 2nd International Conference on Electrical Technology and Automation Engineering 大会官网&#xff1a;http://www.icetae.com/【更多详情】 会议时间&#xff1a;2025年4月25-27日 会议地点&#xff1a…

【弹性计算】弹性计算的技术架构

弹性计算的技术架构 1.工作原理2.总体架构3.控制面4.数据面5.物理设施层 虽然弹性计算的产品种类越来越多&#xff0c;但不同产品的技术架构大同小异。下面以当前最主流的产品形态 —— 云服务器为例&#xff0c;探查其背后的技术秘密。 1.工作原理 云服务器通常以虚拟机的方…

EasyRTC轻量级SDK:智能硬件音视频通信资源的高效利用方案

在智能硬件这片广袤天地里&#xff0c;每一份资源的精打细算都关乎产品的生死存亡。随着物联网技术的疾速演进&#xff0c;实时音视频通信功能已成为众多设备的标配。然而&#xff0c;硬件资源的捉襟见肘&#xff0c;让开发者们常常陷入两难境地。EasyRTC&#xff0c;以它的极致…

Linux | 进程相关概念(进程、进程状态、进程优先级、环境变量、进程地址空间)

文章目录 进程概念1、冯诺依曼体系结构2、进程2.1基本概念2.2描述进程-PCB2.3组织进程2.4查看进程2.5通过系统调用获取进程标识符2.6通过系统调用创建进程-fork初识fork の 头文件与返回值fork函数的调用逻辑和底层逻辑 3、进程状态3.1状态3.2进程状态查看命令3.2.1 ps命令3.2.…

【ESP32接入国产大模型之Deepseek】

【ESP32接入国产大模型之Deepseek】 1. Deepseek大模型1.1 了解Deepseek api1.2 Http接口鉴权1.3. 接口参数说明1.3.1 请求体(request)参数1.3.2 模型推理 2. 先决条件2.1 环境配置2.2 所需零件 3. 核心代码3.1 源码分享3.2 源码解析3.3 连续对话修改后的代码代码说明示例输出注…

OSI 参考模型和 TCP/IP 参考模型

数据通信是很复杂的&#xff0c;很难在一个协议中完成所有功能。因此在制定协议时经常采用的思路是将复杂的数据通信功能由若干协议分别完成&#xff0c;然后将这些协议按照一定的方式组织起来。最典型的是采用分层的方式来组织协议&#xff0c;每一层都有一套清晰明确的功能和…

C# CultureInfo 地区影响字符串

问题 线上遇到有玩家资源加载异常&#xff0c;发现资源路径出现异常字符&#xff1a; 发现是土耳其语下字符串转小写不符合预期&#xff1a; "I".ToLower() -> ı 解决方案 String.ToLower 改成 String.ToLowerInvariant 全局修改禁用文化差异&#xff1a;ht…

蓝桥与力扣刷题(108 将有序数组转换成二叉搜索树)

题目&#xff1a;给你一个整数数组 nums &#xff0c;其中元素已经按 升序 排列&#xff0c;请你将其转换为一棵 平衡二叉搜索树。 示例 1&#xff1a; 输入&#xff1a;nums [-10,-3,0,5,9] 输出&#xff1a;[0,-3,9,-10,null,5] 解释&#xff1a;[0,-10,5,null,-3,null,9]…

python学opencv|读取图像(六十二)使用cv2.morphologyEx()形态学函数实现图像梯度处理

【1】引言 前序已经学习了腐蚀和膨胀的单独作用函数&#xff0c;还研究了按照不同顺序调用腐蚀和膨胀函数调整图像效果&#xff0c;相关文章包括且不限于&#xff1a; python学opencv|读取图像&#xff08;六十一&#xff09;先后使用cv2.dilate()函数和cv2.erode()函数实现图…

(萌新入门)如何从起步阶段开始学习STM32 —— 0.碎碎念

目录 前言与导论 碎碎念 所以&#xff0c;我到底需要知道哪些东西呢 从一些基础的概念入手 常见的工具和说法 ST公司 MDK5 (Keil5) CubeMX 如何使用MDK5的一些常用功能 MDK5的一些常见的设置 前言与导论 非常感谢2301_77816627-CSDN博客的提问&#xff0c;他非常好奇…

线程池-抢票系统性能优化

文章目录 引言-购票系统线程池购票系统-线程池优化 池化 vs 未池化 引言-购票系统 public class App implements Runnable {private static int tickets 100;private static int users 10000;private final ReentrantLock lock new ReentrantLock(true);public void run() …

soular基础教程-使用指南

soular是TikLab DevOps工具链的统一帐号中心&#xff0c;今天来介绍如何使用 soular 配置你的组织、工作台&#xff0c;快速入门上手。 &#xfeff; 1. 账号管理 可以对账号信息进行多方面管理&#xff0c;包括分配不同的部门、用户组等&#xff0c;从而确保账号权限和职责…

大数据SQL调优专题——Hive执行原理

引入 Apache Hive 是基于Hadoop的数据仓库工具&#xff0c;它可以使用SQL来读取、写入和管理存在分布式文件系统中的海量数据。在Hive中&#xff0c;HQL默认转换成MapReduce程序运行到Yarn集群中&#xff0c;大大降低了非Java开发者数据分析的门槛&#xff0c;并且Hive提供命令…

细胞计数专题 | LUNA-FX7™新自动对焦算法提高极低细胞浓度下的细胞计数准确性

现代细胞计数仪采用自动化方法&#xff0c;在特定浓度范围内进行细胞计数。其上限受限于在高浓度条件下准确区分细胞边界的能力&#xff0c;而相机视野等因素则决定了下限。在图像中仅包含少量可识别细胞或特征的情况下&#xff0c;自动对焦可能会失效&#xff0c;从而影响细胞…