# NAME GIS::Distance - Calculate geographic distances. # SYNOPSIS ```perl 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"](#formulas) are available that provide different levels of accuracy versus speed. [GIS::Distance::Fast](https://metacpan.org/pod/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 ```perl 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](https://metacpan.org/pod/Geo::Point) objects. Returns a [Class::Measure::Length](https://metacpan.org/pod/Class::Measure::Length) object for the distance between the two degree lats/lons. See ["distance\_metal"](#distance_metal) for a faster, but less feature rich, method. ## distance\_metal This works just like ["distance"](#distance) except for: - Does not accept [Geo::Point](https://metacpan.org/pod/Geo::Point) objects. Only decimal latitude and longitude pairs. - Does not return a [Class::Measure](https://metacpan.org/pod/Class::Measure) object. Instead kilometers are always returned. - Does no argument checking. - Does not support ["formula\_args"](#formula_args), which are supported by at least the [GIS::Distance::GeoEllipsoid](https://metacpan.org/pod/GIS::Distance::GeoEllipsoid) formula. Read more in the ["SPEED"](#speed) section. Calling this gets you pretty close to the fastest bare metal speed you can get. The speed improvements of calling this is noticeable over hundreds of thousands of iterations only and you've got to decide if its worth the safety and features you are dropping. # ARGUMENTS ```perl my $gis = GIS::Distance->new( $formula, @formula_args ); ``` When you call `GIS::Distance-`new()> you may pass a formula for ["formula"](#formula) and any additional arguments will be slurped into ["formula\_args"](#formula_args). # ATTRIBUTES ## formula ``` print $gis->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: ```perl 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](https://metacpan.org/pod/GIS::Distance::Fast) and they will be automatically used. You may globally disable the automatic use of the `Fast::` formulas by setting the `GIS_DISTANCE_PP` environment variable. Although, its likely simpler to just provide a full module name of a formula to get the same effect: ```perl my $gis = GIS::Distance->new( 'GIS::Distance::Haversine' ); ``` ## formula\_args Returns the formula arguments, an array ref, containing the rest of the arguments passed to `new()` (anything passed after the ["formula"](#formula)). Most formulas do not take arguments. If they do it will be described in their respective documentation. ## formula\_module Returns the fully qualified module name that ["formula"](#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](https://metacpan.org/pod/GIS::Distance::Fast) to get the XS variants for most of the PP formulas. Use ["distance\_metal"](#distance_metal) instead of ["distance"](#distance). Call the undocumented `distance()` function that each formula module has. For example you could bypass this module entirely and just do: ```perl 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 a benchmarks for these options: ``` 2019-03-13T09:34:00Z GIS::Distance 0.15 GIS::Distance::Fast 0.12 GIS::Distance::Fast::Haversine 0.12 GIS::Distance::Haversine 0.15 Rate PP Haversine - GIS::Distance->distance 123213/s XS Haversine - GIS::Distance->distance 196232/s PP Haversine - GIS::Distance->distance_metal 356379/s PP Haversine - GIS::Distance::Haversine::distance 385208/s XS Haversine - GIS::Distance->distance_metal 3205128/s XS Haversine - GIS::Distance::Fast::Haversine::distance 8620690/s ``` You can run your own benchmarks using the included `author/bench` script. The above results were produced with: ``` author/bench -f Haversine ``` The slowest result was about `125000/s`, or about `8ms` each call. This could be a substantial burden in some contexts, such as live HTTP responses to human users and running large batch jobs, to name just two. In conclusion, if you can justify the speed gain, switching to ["distance\_metal"](#distance_metal) and installing [GIS::Distance::Fast](https://metacpan.org/pod/GIS::Distance::Fast) looks to be an ideal setup. As always, YMMV. # COORDINATES When passing latitudinal and longitudinal coordinates to ["distance"](#distance) they must always be in decimal degree format. Here is some sample code for converting from other formats to decimal: ```perl # 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](https://metacpan.org/pod/Math::Trig)'s rad2deg function. # FORMULAS These formulas come bundled with this distribution: - [GIS::Distance::ALT](https://metacpan.org/pod/GIS::Distance::ALT) - [GIS::Distance::Cosine](https://metacpan.org/pod/GIS::Distance::Cosine) - [GIS::Distance::GreatCircle](https://metacpan.org/pod/GIS::Distance::GreatCircle) - [GIS::Distance::Haversine](https://metacpan.org/pod/GIS::Distance::Haversine) - [GIS::Distance::MathTrig](https://metacpan.org/pod/GIS::Distance::MathTrig) - [GIS::Distance::Null](https://metacpan.org/pod/GIS::Distance::Null) - [GIS::Distance::Polar](https://metacpan.org/pod/GIS::Distance::Polar) - [GIS::Distance::Vincenty](https://metacpan.org/pod/GIS::Distance::Vincenty) These formulas are available on CPAN: - ["FORMULAS" in GIS::Distance::Fast](https://metacpan.org/pod/GIS::Distance::Fast#FORMULAS) - [GIS::Distance::GeoEllipsoid](https://metacpan.org/pod/GIS::Distance::GeoEllipsoid) # SEE ALSO - [Geo::Distance](https://metacpan.org/pod/Geo::Distance) - Is deprecated in favor of using this module. - [Geo::Distance::Google](https://metacpan.org/pod/Geo::Distance::Google) - While in the Geo::Distance namespace, this isn't actually related to Geo::Distance at all. Might be useful though. - [GIS::Distance::Lite](https://metacpan.org/pod/GIS::Distance::Lite) - An old fork of this module, not recommended. - [Geo::Distance::XS](https://metacpan.org/pod/Geo::Distance::XS) - Used to be used by [Geo::Distance](https://metacpan.org/pod/Geo::Distance) but no longer is. - [Geo::Ellipsoid](https://metacpan.org/pod/Geo::Ellipsoid) - Or use [GIS::Distance::GeoEllipsoid](https://metacpan.org/pod/GIS::Distance::GeoEllipsoid) for a uniform interface. - [Geo::Inverse](https://metacpan.org/pod/Geo::Inverse) - Does some distance calculations, but seems less than useful as all the code looks to be taken from [Geo::Ellipsoid](https://metacpan.org/pod/Geo::Ellipsoid). # SUPPORT Please submit bugs and feature requests to the GIS-Distance GitHub issue tracker: [https://github.com/bluefeet/GIS-Distance/issues](https://github.com/bluefeet/GIS-Distance/issues) # AUTHORS ``` Aran Clary Deltac ``` # LICENSE This library is free software; you can redistribute it and/or modify it under the same terms as Perl itself.