>백엔드 개발 >C#.Net 튜토리얼 >KD 트리를 통해 가장 가까운 지점을 검색하는 C# 예제 분석

KD 트리를 통해 가장 가까운 지점을 검색하는 C# 예제 분석

黄舟
黄舟원래의
2017-10-05 15:32:022100검색

이 글에서는 KD 트리를 통해 가장 가까운 지점을 찾는 C#을 주로 소개합니다. 관심 있는 친구들은 참고해 보세요.

이 글에서는 먼저 Kd-Tree의 구성 방법을 소개하고 검색 과정을 소개합니다. Kd-Tree의 코드 구현, 그리고 마지막으로 C# 언어를 사용하여 제가 구현한 2차원 KD 트리 코드입니다. 이것은 제가 직접 구현한 최초의 트리 형태의 데이터 구조이기도 합니다. 필연적으로 이해의 차이가 있을 수 있으니 정정해 주시기 바랍니다.

1. KD 트리 소개

K-차원 트리라고도 알려진 Kd-Tree(KD 트리)는 대규모 최근접 이웃 분석을 수행하는 데 자주 사용되는 고차원 인덱스 트리 데이터 구조입니다. -고차원 데이터 공간을 검색하고 가장 가까운 이웃 검색을 근사화합니다. 제가 구현한 KD 트리는 2차원 Kd-트리입니다. 목적은 포인트 세트에서 가장 가까운 포인트를 찾는 것입니다. 참고 자료는 Kd-Tree의 Baidu Encyclopedia입니다. 그리고 코드는 Baidu Encyclopedia의 논리에 따라 구성됩니다.

2. KD 트리의 수학적 설명

3. KD 트리 구성 방법

다음은 Kd-트리 구성에 사용되는 2차원 점 집합입니다. 입체감도 비슷해요.
트리에 있는 각 노드의 데이터 유형:


public class KDTreeNode
  {
    /// <summary>
    /// 分裂点
    /// </summary>
    public Point pisionPoint { get; set; }

    /// <summary>
    /// 分裂类型
    /// </summary>
    public EnumpisionType pisionType { get; set; }

    /// <summary>
    /// 左子节点
    /// </summary>
    public KDTreeNode LeftChild { get; set; }

    /// <summary>
    /// 右子节点
    /// </summary>
    public KDTreeNode RightChild { get; set; }
  }

3.1 KD 트리 구성 논리 프로세스

  • 모든 점을 집합 a에 넣습니다.

  • 집합에 있는 모든 점의 X 좌표의 분산 xv를 계산합니다. , Y 좌표는 분산 yv

  • 를 찾습니다. xv > yv이면 X 좌표에 따라 집합 a를 정렬합니다. yv > xv이면 집합 a는 y 좌표에 따라 정렬됩니다.

  • 정렬된 집합 a의 중앙값 m을 구합니다. 그런 다음 m을 중단점으로 사용하고 [0,m-2]로 인덱스된 점을 a1 세트에 넣습니다. [m,a.count]로 인덱스된 포인트를 a2 집합에 넣습니다(포인트 m의 인덱스는 m-1입니다).

  • 노드를 구성합니다. 노드의 값은 a[m-1]입니다. 작업 집합의 노드 수가 1보다 크면 왼쪽 노드는 [0, m에 대해 2~5단계를 반복합니다. -2], 오른쪽 노드는 [m,a.count]입니다. 그렇지 않으면 노드는 리프 노드입니다.

3.2 코드 구현


private KDTreeNode CreateTreeNode(List<Point> pointList)
{
  if (pointList.Count > 0)
  {
    // 计算方差
    double xObtainVariance = ObtainVariance(CreateXList(pointList));
    double yObtainVariance = ObtainVariance(CreateYList(pointList));

    // 根据方差确定分裂维度
    EnumpisionType pisionType = SortListByXOrYVariances(xObtainVariance,    yObtainVariance, ref pointList);

    // 获得中位数
    Point medianPoint = ObtainMedian(pointList);
    int medianIndex = pointList.Count / 2;

    // 构建节点
    KDTreeNode treeNode = new KDTreeNode()
    {
      pisionPoint = medianPoint,
      pisionType = pisionType,
      LeftChild = CreateTreeNode(pointList.Take(medianIndex).ToList()),
      RightChild = CreateTreeNode(pointList.Skip(medianIndex + 1).ToList())
    };
    return treeNode;
  }
  else
  {
    return null;
  }
}

4. KD 트리 검색 방법

Kd-Tree의 전체 검색 과정은 먼저 일반 검색을 기반으로 가장 가까운 리프 노드를 찾습니다. 하지만 이 리프 노드가 반드시 가장 가까운 지점은 아닙니다. 그런 다음 역추적 작업을 수행하여 가장 가까운 지점을 찾습니다.

4.1 KD 트리 검색 논리 프로세스

  • 포인트 세트를 기반으로 구축된 트리 t와 검색 포인트 p에 대해 루트 노드를 노드 t로 사용하여 다음 작업을 수행합니다.

  • t가 리프 노드. 그런 다음 가장 가까운 점 n의 값이 t인 분할 점의 값을 얻고, t가 리프 노드가 아니면 3단계로 진행합니다.

  • 그런 다음 t의 분할 방법을 결정합니다. x축을 따라 분할됩니다. 그런 다음 p의 x 값을 노드 분할 지점의 x 값과 비교합니다. 그렇지 않으면 Y 좌표를 비교합니다.

  • p의 비교 값이 t의 비교 값보다 작으면 t는 t의 왼쪽 자식 노드로 지정됩니다. 그렇지 않으면 t를 t의 오른쪽 하위 노드로 지정하고 2

  • 단계를 수행하여 검색 지점 m을 정의하고 m을 n

  • 으로 설정하고 m과 p 사이의 거리 d1과 n 사이의 거리 d2를 계산합니다. 그리고 m.

  • d1 >= d2이고 상위 노드가 있는 경우 m의 상위 노드를 m의 값으로 사용하고 상위 노드가 없으면 d1인 경우 실제 가장 가까운 지점 TN을 가져옵니다. < d2는 n을 의미합니다. 해당 지점이 가장 가까운 지점이 아닙니다. n에 형제 노드가 있으면 n = n의 형제 노드이고, n에 형제 노드가 없으면 n = n의 상위 노드입니다. 원래 n개 노드를 삭제합니다. m 값을 새로운 n 노드로 설정합니다. 6단계를 수행합니다.


  • 4.2 코드 구현

public Point FindNearest(Point searchPoint)
{
  // 按照查找方式寻找最近点
  Point nearestPoint = DFSSearch(this.rootNode, searchPoint);
  
  // 进行回溯
  return BacktrcakSearch(searchPoint, nearestPoint);
}


private Point DFSSearch(KDTreeNode node,Point searchPoint,bool pushStack = true)
{
  if(pushStack == true)
  {
    // 利用堆栈记录查询的路径,由于树节点中没有记载父节点的原因
    backtrackStack.Push(node);
  }
  if (node.pisionType == EnumpisionType.X)
  {
    return DFSXsearch(node,searchPoint);
  }
  else
  {
    return DFSYsearch(node, searchPoint);
  }
}

private Point BacktrcakSearch(Point searchPoint,Point nearestPoint)
{
  // 如果记录路径的堆栈为空则表示已经回溯到根节点,则查到的最近点就是真正的最近点
  if (backtrackStack.IsEmpty())
  {
    return nearestPoint;
  }
  else
  {
    KDTreeNode trackNode = backtrackStack.Pop();
    
    // 分别求回溯点与最近点距查找点的距离
    double backtrackDistance = ObtainDistanFromTwoPoint(searchPoint,     trackNode.pisionPoint);
    double nearestPointDistance = ObtainDistanFromTwoPoint(searchPoint, nearestPoint);
    
    if (backtrackDistance < nearestPointDistance)
    {
      // 深拷贝节点的目的是为了避免损坏树
      KDTreeNode searchNode = new KDTreeNode()
      {
        pisionPoint = trackNode.pisionPoint,
        pisionType = trackNode.pisionType,
        LeftChild = trackNode.LeftChild,
        RightChild = trackNode.RightChild
      };
      nearestPoint = DFSBackTrackingSearch(searchNode, searchPoint);
   }
   // 递归到根节点
   return BacktrcakSearch(searchPoint, nearestPoint);
  }
}

위 내용은 KD 트리를 통해 가장 가까운 지점을 검색하는 C# 예제 분석의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.