Map Matching
kdtree.cpp
Go to the documentation of this file.
1 #include "kdtree.h"
2 
3 #include <algorithm>
4 #include <limits>
5 
6 #define MAX_DOUBLE std::numeric_limits<double>::max()
7 
8 const double KDTree::BIG(1.0e99);
9 
10 KDTree::KDTree(std::vector<Point>& points)
11  : m_vectorOfPoints(points)
12  , m_noOfPoints(points.size())
13  , m_vectorOfPointIndexes(m_noOfPoints)
14  , m_vectorOfReversedPointIndexes(m_noOfPoints)
15 {
16  int ntmp, m, k, kk, j, nowtask, jbox, np, tmom, tdim, ptlo, pthi;
17  int* hp;
18  double* cp;
19  int taskmom[50], taskdim[50];
20  for (k = 0; k < m_noOfPoints; k++)
21  m_vectorOfPointIndexes[k] = k; // initialize index of points
22 
23  // calculate the number of boxes and allocate memory
24  m = 1;
25  for (ntmp = m_noOfPoints; ntmp; ntmp >>= 1)
26  m <<= 1; // at each depth, the numlber of boxes is doubled
27  m_noOfBoxes = 2 * m_noOfPoints - (m >> 1);
28  if (m < m_noOfBoxes)
29  m_noOfBoxes = m;
30  m_noOfBoxes--; // account for mother box
32 
33  // copy the points coordinates into a contiguous array
34  m_arrayOfCoordinates = new double[2 * m_noOfPoints];
35  for (j = 0, kk = 0; j < 2; j++, kk += m_noOfPoints) { // j allows access to x and y (point in 2 dimensions)
36  for (k = 0; k < m_noOfPoints; k++)
37  m_arrayOfCoordinates[kk + k] = m_vectorOfPoints[k].x(j);
38  }
39 
40  // initialize the root box and put it on the task list for subdivision
41  Point low(0.0, 0.0), high(MAX_DOUBLE, MAX_DOUBLE);
42  m_arrayOfBoxes[0] = BoxNode(low, high, 0, 0, 0, 0, m_noOfPoints - 1);
43  jbox = 0;
44  taskmom[1] = 0;
45  taskdim[1] = 0;
46  nowtask = 1;
47 
48  while (nowtask) { // main loop over pending task
49  tmom = taskmom[nowtask];
50  tdim = taskdim[nowtask];
51  --nowtask;
54  hp = &m_vectorOfPointIndexes[ptlo]; // points to left end of subdivision
55  cp = &m_arrayOfCoordinates[tdim * m_noOfPoints]; // points to coordinate list for current dimension
56  np = pthi - ptlo + 1; // number of points in the subdivision
57  kk = (np - 1) / 2; // index of the last point on left (boundary point)
58 
59  (void)partition(kk, hp, np, cp);
60  high = m_arrayOfBoxes[tmom].m_high;
61  low = m_arrayOfBoxes[tmom].m_low;
62  high.setx(tdim, m_arrayOfCoordinates[tdim * m_noOfPoints + hp[kk]]);
63  low.setx(tdim, m_arrayOfCoordinates[tdim * m_noOfPoints + hp[kk]]);
64  m_arrayOfBoxes[++jbox] = BoxNode(m_arrayOfBoxes[tmom].m_low, high, tmom, 0, 0, ptlo, ptlo + kk);
65  m_arrayOfBoxes[++jbox] = BoxNode(low, m_arrayOfBoxes[tmom].m_high, tmom, 0, 0, ptlo + kk + 1, pthi);
66  m_arrayOfBoxes[tmom].m_daughterBox1 = jbox - 1;
67  m_arrayOfBoxes[tmom].m_daughterBox2 = jbox;
68 
69  if (kk > 1) {
70  taskmom[++nowtask] = jbox - 1;
71  taskdim[nowtask] = (tdim + 1) % 2;
72  }
73  if (np - kk > 3) {
74  taskmom[++nowtask] = jbox;
75  taskdim[nowtask] = (tdim + 1) % 2;
76  }
77  //nowtask= 0;
78  }
79  for (j = 0; j < m_noOfPoints; j++)
81  delete[] m_arrayOfCoordinates;
82 }
83 
85 {
86  delete[] m_arrayOfBoxes; // TODO check if it works
87  delete[] m_arrayOfCoordinates;
88 }
89 
90 int KDTree::partition(const int k, int* idx, int n, double* arr)
91 {
92  int i, ia, ir, j, l, mid;
93  double a;
94 
95  l = 0;
96  ir = n - 1;
97  for (;;) {
98  if (ir <= l + 1) {
99  if (ir == l + 1 && arr[idx[ir]] < arr[idx[l]]) {
100  std::swap(idx[l], idx[ir]);
101  }
102  return idx[k];
103  } else {
104  mid = (l + ir) >> 1;
105  std::swap(idx[mid], idx[l + 1]);
106  if (arr[idx[l]] > arr[idx[ir]])
107  std::swap(idx[l], idx[ir]);
108  if (arr[idx[l + 1]] > arr[idx[ir]])
109  std::swap(idx[l + 1], idx[ir]);
110  if (arr[idx[l]] > arr[idx[l + 1]])
111  std::swap(idx[l], idx[l + 1]);
112  i = l + 1;
113  j = ir;
114  ia = idx[l + 1];
115  a = arr[ia];
116  for (;;) {
117  do
118  i++;
119  while (arr[idx[i]] < a);
120  do
121  j--;
122  while (arr[idx[j]] > a);
123  if (j < i)
124  break;
125  std::swap(idx[i], idx[j]);
126  }
127  idx[l + 1] = idx[j];
128  idx[j] = ia;
129  if (j >= k)
130  ir = j - 1;
131  if (j <= k)
132  l = i;
133  }
134  }
135 }
136 
137 double KDTree::distance(int idxOfPoint1, int idxOfPoint2)
138 {
139  if (idxOfPoint1 == idxOfPoint2)
140  return BIG;
141  else
142  return m_vectorOfPoints[idxOfPoint1].distanceToPoint(m_vectorOfPoints[idxOfPoint2]);
143 }
144 
145 int KDTree::locate(const Point& p)
146 {
147  int nb, d1, jdim;
148  nb = 0;
149  jdim = 0; // start with the root box
150  while (m_arrayOfBoxes[nb].m_daughterBox1) { // as far as possible down the tree
152  if (p.x(jdim) <= m_arrayOfBoxes[d1].m_high.x(jdim))
153  nb = d1;
154  else
156  ++jdim;
157  jdim %= 2; // increment dimension cyclically
158  }
159  return nb;
160 }
Point m_high
Definition: box.h:18
Definition: boxnode.h:6
void setx(double x)
Definition: point.cpp:80
static const double BIG
Definition: kdtree.h:11
int m_daughterBox2
Definition: boxnode.h:22
std::vector< int > m_vectorOfPointIndexes
Definition: kdtree.h:25
int m_noOfPoints
Definition: kdtree.h:23
std::vector< Point > m_vectorOfPoints
Definition: kdtree.h:22
int m_indexOfLowerPoint
Definition: boxnode.h:23
int partition(const int k, int *index, int n, double *arr)
Definition: kdtree.cpp:90
int m_noOfBoxes
Definition: kdtree.h:21
int m_indexOfUpperPoint
Definition: boxnode.h:23
Point m_low
Definition: box.h:17
double x() const
Definition: point.cpp:75
int locate(const Point &p)
Definition: kdtree.cpp:145
BoxNode * m_arrayOfBoxes
Definition: kdtree.h:24
KDTree(std::vector< Point > &points)
Definition: kdtree.cpp:10
double * m_arrayOfCoordinates
Definition: kdtree.h:26
#define MAX_DOUBLE
Definition: kdtree.cpp:6
int m_daughterBox1
Definition: boxnode.h:22
~KDTree()
Definition: kdtree.cpp:84
double distance(int idxOfPoint1, int idxOfPoint2)
Definition: kdtree.cpp:137
The Point class.
Definition: point.h:20
std::vector< int > m_vectorOfReversedPointIndexes
Definition: kdtree.h:25