diff --git a/rtree.go b/rtree.go index 5dee5a7..0f3a82a 100644 --- a/rtree.go +++ b/rtree.go @@ -765,7 +765,8 @@ func (tree *Rtree) nearestNeighbor(p Point, n *node, d float64, nearest Spatial) for _, e := range n.entries { minDist := p.minDist(e.bb) - if minDist > minMinMaxDist { + // Add a bit of tolerance to guard against floating point rounding errors. + if minDist > minMinMaxDist+1e-6 { continue } diff --git a/rtree_test.go b/rtree_test.go index 7a1f860..f280aa3 100644 --- a/rtree_test.go +++ b/rtree_test.go @@ -1354,6 +1354,24 @@ func TestNearestNeighborsHalf(t *testing.T) { } } +func TestMinMaxDistFloatingPointRoundingError(t *testing.T) { + rects := []Rect{ + Point{1134900, 15600}.ToRect(0), + Point{1134900, 25600}.ToRect(0), + Point{1134900, 22805}.ToRect(0), + Point{1134900, 29116}.ToRect(0), + } + things := make([]Spatial, 0, len(rects)) + for i := range rects { + things = append(things, &rects[i]) + } + rt := NewTree(2, 1, 2, things...) + n := rt.NearestNeighbor(Point{1134851.8, 25570.8}) + if n != things[1] { + t.Fatalf("wrong neighbor, expected %v, got %v", things[1], n) + } +} + func ensureOrderedSubset(t *testing.T, actual []Spatial, expected []Spatial) { for i := range actual { if len(expected)-1 < i || actual[i] != expected[i] {