mirror of https://github.com/apache/lucene.git
LUCENE-7185: fix buggy worst-case error and add test for absurd distances
This commit is contained in:
parent
4ac2546e5b
commit
b839325740
|
@ -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.
|
||||||
*
|
*
|
||||||
|
|
|
@ -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();
|
||||||
|
|
Loading…
Reference in New Issue