动态增删kdtree(ikdtree)主要思路

2023-10-27 02:15

本文主要是介绍动态增删kdtree(ikdtree)主要思路,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

ikdtree本质上也是一种kdtree,基本的构造方法和kdtree是一样的,本文主要记录两者不一样的地方,以港大MaRS实验室最新开源的增量式 kd-tree(https://github.com/hku-mars/ikd-Tree)里面的一些代码作为示范。

以下是ikdtree结构体包括的基本变量,在港大开源的代码中结构变量是远远多于这里的,主要是为了利用其他方面(例如多线程)对树进一步优化加速。

{

     // 寻常kdtree也有的:
        uint8_t division_axis;  //以哪一个轴(x,y,z)作为的切分平面法线

        KD_TREE_NODE *left_son_ptr = nullptr; //左孩子节点
        KD_TREE_NODE *right_son_ptr = nullptr;//右孩子节点

        float node_range_x[2], node_range_y[2], node_range_z[2];   //AABB包围盒

 

    // 以下主要是用于增删的变量:

       PointType point; //点坐标
        int TreeSize = 1; //包括它本身和它左右孩子的总数量
        int invalid_point_num = 0; //包括自己在内已经被标记删除点的数量
        bool point_deleted = false; //该节点是否被标记删除
        KD_TREE_NODE *father_ptr = nullptr; //它的父亲节点

}

首先 说一下这里的点坐标,虽然有些kdtree的写法上也在内部节点上保存了点坐标,但更通常的做法是仅在叶子节点上有存储,而在ikdtree中,这个变量内部节点则必须存在,这在动态点插入的时候有很大的作用。

ikdtree节点的删除操作采用的是一种延迟删除操作的做法,这种做法其实并不陌生,c++ 标准库中remove就是这种策略,即仅对节点做删除标记,而不立即更新删除树,在结构体中point_deleted就是起到这个作用,标记完后,AABB盒是要立即更新的,但是要排除已经标记删除的数据。

ikdtree的插入操作是直接将节点插入在树的末尾,当然这也是有规则的,在港大开源的代码中,插入节点满足大于左节点(这里点坐标就有了用处),不满足则直接遍历右节点从而放在右节点末端,全部放在右节点末端是为了人为破坏树的平衡性,这是后面判断树更新的一个重要依据。同删除一样,插入完毕后,AABB盒要立即更新。

6521c3261f704ad691ae02e878afe354.jpeg

 上述做完之后,可以认为已经动态改变了,但是也可以看见,此时的删除和插入操作其实已经破坏了kdtree的特性,例如删除操作破坏了平衡特性,增加了很多无效的节点,增加了遍历,插入操作则破坏了树平衡性和有序性两个特性(这里注明一下,有序性的破坏并不会影响结果,这里让我迷糊了很久,因为寻找最近点即使有序的二叉树仅通过包围盒也是无法排除另一个孩子树的)。在破坏较小时,相比于重新建立树的效率相比,些许的效率损失可以接受,但是当变化太大后,这个tree结构也就几乎没有了加速的效果。因此在ikdtree中,在认为已经破坏过大时,必须更新树结构。ikdtree中提出了两个依据来判断是否树已经破坏过大:

一是被标记删除的个数占整个节点数量的比例,港大源代码中默认是要小于0.5,二是不平衡性,即是单个孩子节点数量占整个节点数量的比例,港大源代码中默认为在0.3~0.7之间为合理的。

当任何一个依据不满足时,树就要立即更新。

e8cb5be632954f699a91c0baf7549908.jpeg

在树更新的过程中,也不是整个树都要更新,仅仅只是不满足两个依据的子树进行自我更新,进一步优化了效率,示例如下图。这里结构体中的父亲节点就有了可用之地。

7726d951aeef47a38fbd00dbeb8ad777.png

d9ac3feff423429e97572874f5641eb1.jpeg 

 

这篇关于动态增删kdtree(ikdtree)主要思路的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Spring Boot整合Redis注解实现增删改查功能(Redis注解使用)

《SpringBoot整合Redis注解实现增删改查功能(Redis注解使用)》文章介绍了如何使用SpringBoot整合Redis注解实现增删改查功能,包括配置、实体类、Repository、Se... 目录配置Redis连接定义实体类创建Repository接口增删改查操作示例插入数据查询数据删除数据更

Spring Gateway动态路由实现方案

《SpringGateway动态路由实现方案》本文主要介绍了SpringGateway动态路由实现方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随... 目录前沿何为路由RouteDefinitionRouteLocator工作流程动态路由实现尾巴前沿S

Python动态处理文件编码的完整指南

《Python动态处理文件编码的完整指南》在Python文件处理的高级应用中,我们经常会遇到需要动态处理文件编码的场景,本文将深入探讨Python中动态处理文件编码的技术,有需要的小伙伴可以了解下... 目录引言一、理解python的文件编码体系1.1 Python的IO层次结构1.2 编码问题的常见场景二

Java使用Javassist动态生成HelloWorld类

《Java使用Javassist动态生成HelloWorld类》Javassist是一个非常强大的字节码操作和定义库,它允许开发者在运行时创建新的类或者修改现有的类,本文将简单介绍如何使用Javass... 目录1. Javassist简介2. 环境准备3. 动态生成HelloWorld类3.1 创建CtC

Spring创建Bean的八种主要方式详解

《Spring创建Bean的八种主要方式详解》Spring(尤其是SpringBoot)提供了多种方式来让容器创建和管理Bean,@Component、@Configuration+@Bean、@En... 目录引言一、Spring 创建 Bean 的 8 种主要方式1. @Component 及其衍生注解

Go语言连接MySQL数据库执行基本的增删改查

《Go语言连接MySQL数据库执行基本的增删改查》在后端开发中,MySQL是最常用的关系型数据库之一,本文主要为大家详细介绍了如何使用Go连接MySQL数据库并执行基本的增删改查吧... 目录Go语言连接mysql数据库准备工作安装 MySQL 驱动代码实现运行结果注意事项Go语言执行基本的增删改查准备工作

go动态限制并发数量的实现示例

《go动态限制并发数量的实现示例》本文主要介绍了Go并发控制方法,通过带缓冲通道和第三方库实现并发数量限制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录带有缓冲大小的通道使用第三方库其他控制并发的方法因为go从语言层面支持并发,所以面试百分百会问到

MySql基本查询之表的增删查改+聚合函数案例详解

《MySql基本查询之表的增删查改+聚合函数案例详解》本文详解SQL的CURD操作INSERT用于数据插入(单行/多行及冲突处理),SELECT实现数据检索(列选择、条件过滤、排序分页),UPDATE... 目录一、Create1.1 单行数据 + 全列插入1.2 多行数据 + 指定列插入1.3 插入否则更

Qt使用QSqlDatabase连接MySQL实现增删改查功能

《Qt使用QSqlDatabase连接MySQL实现增删改查功能》这篇文章主要为大家详细介绍了Qt如何使用QSqlDatabase连接MySQL实现增删改查功能,文中的示例代码讲解详细,感兴趣的小伙伴... 目录一、创建数据表二、连接mysql数据库三、封装成一个完整的轻量级 ORM 风格类3.1 表结构

一文详解SpringBoot中控制器的动态注册与卸载

《一文详解SpringBoot中控制器的动态注册与卸载》在项目开发中,通过动态注册和卸载控制器功能,可以根据业务场景和项目需要实现功能的动态增加、删除,提高系统的灵活性和可扩展性,下面我们就来看看Sp... 目录项目结构1. 创建 Spring Boot 启动类2. 创建一个测试控制器3. 创建动态控制器注