NAME
GIS::Distance - Calculate geographic distances.
SYNOPSIS
use GIS::Distance;
# Use the GIS::Distance::Haversine formula by default:
my $gis = GIS::Distance->new();
# Or choose a different formula:
my $gis = GIS::Distance->new( 'Polar' );
my $distance = $gis->distance( $lat1, $lon1, $lat2, $lon2 );
print $distance->meters();
DESCRIPTION
This module calculates distances between geographic points on, at the moment, planet Earth. Various "FORMULAS" are available that provide different levels of accuracy versus speed.
GIS::Distance::Fast, a separate distribution, ships with C implmentations of some of the formulas shipped with GIS::Distance. If you're looking for speed then install it and the ::Fast formulas will be automatically used by this module.
METHODS
distance
my $distance = $gis->distance( $lat1, $lon1, $lat2, $lon2 );
my $point1 = Geo::Point->latlong( $lat1, $lon1 );
my $point2 = Geo::Point->latlong( $lat2, $lon2 );
my $distance = $gis->distance( $point1, $point2 );
Takes either two decimal latitude and longitude pairs, or two Geo::Point objects.
Returns a Class::Measure::Length object for the distance between the two degree lats/lons.
See "distance_metal" for a faster, but less feature rich, method.
distance_metal
This works just like "distance" except for:
Does not accept Geo::Point objects. Only decimal latitude and longitude pairs.
Does not return a Class::Measure object. Instead kilometers are always returned.
Does no argument checking.
Does not support formula "args", which are needed by at least the GIS::Distance::GeoEllipsoid formula.
Calling this gets you pretty close to the fastest bare metal speed you can get. The speed improvements of calling this is noticeable over millions of iterations only and you've got to decide if its worth the safety and features you are dropping.
ATTRIBUTES
formula
Returns the formula name which was passed as the first argument to new()
.
The formula can be specified as a partial or full module name for that formula. For example, if the formula is set to Haversine
as in:
my $gis = GIS::Distance->new( 'Haversine' );
Then the following modules will be looked for in order:
GIS::Distance::Fast::Haversine
GIS::Distance::Haversine
Haversine
Note that a Fast::
version of the class will be looked for first. By default the Fast::
versions of the formulas, written in C, are not available and the pure perl ones will be used instead. If you would like the Fast::
formulas then install GIS::Distance::Fast and they will be automatically used.
You may disable the automatic use of the Fast::
formulas by setting the GIS_DISTANCE_PP
environment variable.
args
Returns the formula arguments, an array ref, containing the rest of the arguments passed to new()
(anything passed after the "formula"). Most formulas do not take arguments. If they do it will be described in their respective documentation.
module
Returns the fully qualified module name that "formula" resolved to.
SPEED
Not that this module is slow, but if you're doing millions of distance calculations a second you may find that adjusting your code a bit may make it faster. Here are some options.
Install GIS::Distance::Fast to get the XS variants for most of the PP formulas.
Use "distance_metal" instead of "distance".
Call the undocumented distance()
function that each formula module has. For example you could bypass this module entirely and just do:
use GIS::Distance::Fast::Haversine;
my $km = GIS::Distance::Fast::Haversine::distance( @coords );
The above would be the ultimate speed demon (as shown in benchmarking) but throws away some flexibility and adds some foot-gun support.
Here's some benchmarks for these options:
PP Haversine - GIS::Distance->distance 125913/s
XS Haversine - GIS::Distance->distance 203335/s
PP Haversine - GIS::Distance->distance_metal 366569/s
PP Haversine - GIS::Distance::Haversine::distance 390320/s
XS Haversine - GIS::Distance->distance_metal 3289474/s
XS Haversine - GIS::Distance::Fast::Haversine::distance 8064516/s
You can run your own benchmarks using the included author/bench
script. The above results were produced with:
author/bench -f Haversine
Even the slowest result was 125913/s
, which is 125.913/ms
, which means each call took about 0.0079ms
.
In conclusion, if you can justify the speed gain, switching to "distance_metal" and installing GIS::Distance::Fast, seems the ideal setup.
As always, YMMV.
COORDINATES
When passing latitudinal and longitudinal coordinates to "distance" they must always be in decimal degree format. Here is some sample code for converting from other formats to decimal:
# DMS to Decimal
my $decimal = $degrees + ($minutes/60) + ($seconds/3600);
# Precision Six Integer to Decimal
my $decimal = $integer * .000001;
If you want to convert from decimal radians to degrees you can use Math::Trig's rad2deg function.
FORMULAS
These formulas come with this distribution:
These formulas are available on CPAN:
SEE ALSO
Geo::Distance - Is deprecated in favor of using this module.
Geo::Distance::Google - While in the Geo::Distance, namespace this isn't actually related to Geo::Distance at all. Might be useful.
GIS::Distance::Lite - An old fork of this module, not recommended.
Geo::Distance::XS - Used to be used by Geo::Distance but no longer is.
Geo::Ellipsoid - Or use GIS::Distance::GeoEllipsoid for a uniform interface.
Geo::Inverse - Does some distance calculations, but seems less than useful as all the code looks to be taken from Geo::Ellipsoid.
SUPPORT
Please submit bugs and feature requests to the GIS-Distance GitHub issue tracker:
https://github.com/bluefeet/GIS-Distance/issues
AUTHORS
Aran Clary Deltac <bluefeet@gmail.com>
LICENSE
This library is free software; you can redistribute it and/or modify it under the same terms as Perl itself.