LUCENE-7185: fix buggy worst-case error and add test for absurd distances

This commit is contained in:
Robert Muir 2016-04-18 19:47:22 -04:00
parent 4ac2546e5b
commit b839325740
2 changed files with 21 additions and 4 deletions

View File

@ -37,7 +37,7 @@ public class SloppyMath {
* specified in decimal degrees (latitude/longitude). This works correctly * specified in decimal degrees (latitude/longitude). This works correctly
* even if the dateline is between the two points. * even if the dateline is between the two points.
* <p> * <p>
* Error is at most 2E-1 (20cm) from the actual haversine distance, but is typically * Error is at most 4E-1 (40cm) from the actual haversine distance, but is typically
* much smaller for reasonable distances: around 1E-5 (0.01mm) for distances less than * much smaller for reasonable distances: around 1E-5 (0.01mm) for distances less than
* 1000km. * 1000km.
* *

View File

@ -33,7 +33,7 @@ public class TestSloppyMath extends LuceneTestCase {
// accuracy for asin() // accuracy for asin()
static double ASIN_DELTA = 1E-7; static double ASIN_DELTA = 1E-7;
// accuracy for haversinMeters() // accuracy for haversinMeters()
static double HAVERSIN_DELTA = 2E-1; static double HAVERSIN_DELTA = 38E-2;
// accuracy for haversinMeters() for "reasonable" distances (< 1000km) // accuracy for haversinMeters() for "reasonable" distances (< 1000km)
static double REASONABLE_HAVERSIN_DELTA = 1E-5; static double REASONABLE_HAVERSIN_DELTA = 1E-5;
@ -161,12 +161,29 @@ public class TestSloppyMath extends LuceneTestCase {
double lat2 = GeoTestUtil.nextLatitude(); double lat2 = GeoTestUtil.nextLatitude();
double lon2 = GeoTestUtil.nextLongitude(); double lon2 = GeoTestUtil.nextLongitude();
double expected = haversinMeters(lat1, lon1, lat2, lon2); double expected = slowHaversin(lat1, lon1, lat2, lon2);
double actual = slowHaversin(lat1, lon1, lat2, lon2); double actual = haversinMeters(lat1, lon1, lat2, lon2);
assertEquals(expected, actual, HAVERSIN_DELTA); assertEquals(expected, actual, HAVERSIN_DELTA);
} }
} }
/**
* Step across the whole world to find huge absolute errors.
* Don't rely on random number generator to pick these massive distances. */
public void testAcrossWholeWorldSteps() {
for (int lat1 = -90; lat1 <= 90; lat1 += 10) {
for (int lon1 = -180; lon1 <= 180; lon1 += 10) {
for (int lat2 = -90; lat2 <= 90; lat2 += 10) {
for (int lon2 = -180; lon2 <= 180; lon2 += 10) {
double expected = slowHaversin(lat1, lon1, lat2, lon2);
double actual = haversinMeters(lat1, lon1, lat2, lon2);
assertEquals(expected, actual, HAVERSIN_DELTA);
}
}
}
}
}
public void testAgainstSlowVersionReasonable() { public void testAgainstSlowVersionReasonable() {
for (int i = 0; i < 100_000; i++) { for (int i = 0; i < 100_000; i++) {
double lat1 = GeoTestUtil.nextLatitude(); double lat1 = GeoTestUtil.nextLatitude();