KDtree

199 阅读2分钟

1.对KDtree的理解

  首先要知道KDtree的用处,KDtree是用来进行多维数点的,一般这些点都是在在而二维及二维以上,因为一维上的问题,我们基本都可以运用线段树来解决。我对KDtree 的理解就是一个自带剪枝的暴力,并且这个剪枝因为我们对这些多维上的点的较优秀的排列而显得十分有用。

2.前置知识

  在学习KDtree 之前要先知道并会运用西面三个知识点:

​  1) 首先,要会建二叉搜索树,因为整个KDtree 就是一颗二叉搜索树。

  2) 还需要知道什么事估价函数,因为剪枝的时候要运用到估价函数。

​  3) 对空间的想象能力,因为KDtree 是处理图形上的问题,所以还需要有一定的空间想象能力。

3.KDTree的讲解

  因为KDtree 是一种优美的暴力,并且我们要在上面剪枝,所以我们自然想让每一次剪枝,剪下去尽可能大的部分,所以我们能想到每一次将区间等大的分割,既然要的等大的分割,又要是二叉搜索树,我们就要让中间值作为当前节点,所有比它小的都放在它的左面,比它大的都放在它的右面。

  知道大致思路了,就要来定义什么是大小了,因为一个点是在多维里,所以和它有关的值有多个。最好想的就是按读入的顺序,进行排序,第一维作为第一关键字,第二维作为第二关键字,以此类推。我们根据这些点的维度将它们从小到大排序(下面已二维上的点为例),每一次取当前区间的中间值来建树。这样我们就能将整个图分成下面的形式:

  显然这种分法分出的图并不是最有利,因为每一点的管辖范围都太小了。我们考虑另一种分割方式,我们将这些点的排序方式进行改变,我们将排序的关键字每一次向顺时针进行转动,即我们第一次排序的第一关键字是第一维,第二次是第二维……第n

次是第n%维数+1 维。这样上面的图形就可以改变成为:

  这样我们在剪枝的时候就能剪去更多的节点。

  知道了如何去排序,我们现在就要知道怎么来找中间值。在函数库里面有一个函数nth_element ,这个就能实现我们要的功能。这个函数不知道实现的话,可以上网上找一下学习一下。