dart 从LatLng列表计算总距离

jq6vz3qz  于 2023-04-27  发布在  其他
关注(0)|答案(8)|浏览(109)

我使用dart/flutter和'package:latlong/latlong.dart'将一个GPX文件解析成一个LatLng对象列表。这工作正常,但下一步是找到路线的总距离。
现在的问题是
我如何从Dart中的LatLng对象列表中获得总距离?
我以前在Java中使用过半正矢公式,但不确定用Dart实现它的最佳方法。任何建议的帮助都将不胜感激。

编辑

下面是一个基于下面的答案得到总距离的尝试。由于某种原因,它没有添加到totalDistance,也没有返回一个双精度型。
下面的代码假设迭代60个以上的LatLng对象,并找到每个对象之间的距离,添加到循环结束时返回的totalDistance double。

static double getDistanceFromGPSPointsInRoute(List<LatLng> gpsList) {
    double totalDistance = 0.0;

    for (var i = 0; i < gpsList.length; i++) {
      var p = 0.017453292519943295;
      var c = cos;
      var a = 0.5 -
          c((gpsList[i + 1].latitude - gpsList[i].latitude) * p) / 2 +
          c(gpsList[i].latitude * p) *
              c(gpsList[i + 1].latitude * p) *
              (1 - c((gpsList[i + 1].longitude - gpsList[i].longitude) * p)) /
              2;
      double distance = 12742 * asin(sqrt(a));
      totalDistance += distance;
      print('Distance is ${12742 * asin(sqrt(a))}');
    }
    print('Total distance is $totalDistance');
    return totalDistance;
  }

输出

flutter: Distance is 0.004143962775784678
flutter: Distance is 0.0041439635323316775
flutter: Distance is 0.007796918986828574
flutter: Distance is 0.007285385943437824
flutter: Distance is 0.006890844300938902
flutter: Distance is 0.006353952460010352
flutter: Distance is 0.005560051252981138

正如你在上面看到的,没有提到总距离。

LatLng列表示例

flutter: -36.84975 , 174.646685
flutter: -36.849692 , 174.646497
flutter: -36.84967 , 174.646436
flutter: -36.849578 , 174.646264
flutter: -36.849502 , 174.646164
flutter: -36.849367 , 174.646038
flutter: -36.849209 , 174.645959
flutter: -36.849155 , 174.64594
flutter: -36.849107 , 174.645932
flutter: -36.849058 , 174.645922
flutter: -36.848952 , 174.645895
flutter: -36.84886 , 174.645906
flutter: -36.84879 , 174.645913
flutter: -36.848748 , 174.645836
flutter: -36.848744 , 174.645802
emeijp43

emeijp431#

请试试这个。我用谷歌Map测试了它,效果很准确。你可以做一个循环,每次使用2个点,以KM为单位计算总距离。* 我添加了一些随机的虚拟数据来展示它是如何工作的。* 将此代码复制到https://dartpad.dartlang.org/并轻松测试。

import 'dart:math' show cos, sqrt, asin;

void main() {
  double calculateDistance(lat1, lon1, lat2, lon2){
    var p = 0.017453292519943295;
    var c = cos;
    var a = 0.5 - c((lat2 - lat1) * p)/2 + 
          c(lat1 * p) * c(lat2 * p) * 
          (1 - c((lon2 - lon1) * p))/2;
    return 12742 * asin(sqrt(a));
  }
  
  List<dynamic> data = [
    {
      "lat": 44.968046,
      "lng": -94.420307
    },{
      "lat": 44.33328,
      "lng": -89.132008
    },{
      "lat": 33.755787,
      "lng": -116.359998
    },{
      "lat": 33.844843,
      "lng": -116.54911
    },{
      "lat": 44.92057,
      "lng": -93.44786
    },{
      "lat": 44.240309,
      "lng": -91.493619
    },{
      "lat": 44.968041,
      "lng": -94.419696
    },{
      "lat": 44.333304,
      "lng": -89.132027
    },{
      "lat": 33.755783,
      "lng": -116.360066
    },{
      "lat": 33.844847,
      "lng": -116.549069
    },
  ];
  double totalDistance = 0;
  for(var i = 0; i < data.length-1; i++){
    totalDistance += calculateDistance(data[i]["lat"], data[i]["lng"], data[i+1]["lat"], data[i+1]["lng"]);
  }
  print(totalDistance);
}
piv4azn7

piv4azn72#

我用了westdabestdb的答案,没关系。
你也可以使用geolocator插件。

Geolocator.distanceBetween(lat1, lon1, lat2, lon2);

geolocator doc
或者你可以用这种方式without using plugin

wsewodh2

wsewodh23#

使用latlong库进行常见的纬度和经度计算。这支持“Haversine”和“Vincenty”算法。

List<dynamic> data = [
    {
        "lat": 44.968046,
        "lng": -94.420307
    },
    ...
];

final Distance distance = Distance();
double totalDistanceInM = 0;
double totalDistanceInKm = 0;

for(var i = 0; i < data.length - 1; i++){
    totalDistanceInM += distance(
        LatLng(data[i]["lat"], data[i]["lng"]),
        LatLng(data[i+1]["lat"], data[i+1]["lng"])
    );

    totalDistanceInKm += distance.as(
        LengthUnit.Kilometer, 
        LatLng(data[i]["lat"], data[i]["lng"]), 
        LatLng(data[i+1]["lat"], data[i+1]["lng"]),
    );
}

更新

使用latlong2来代替引入空安全性,正如@Mehul Pamale在评论中所说的那样。

du7egjpx

du7egjpx4#

此函数计算两点之间的距离(给定这些点的纬度/经度)。

unit = the unit you desire for results              
      where: 'M' is statute miles (default) 
             'K' is kilometers            
             'N' is nautical miles
String distance(
          double lat1, double lon1, double lat2, double lon2, String unit) {
         double theta = lon1 - lon2;
         double dist = sin(deg2rad(lat1)) * sin(deg2rad(lat2)) +
         cos(deg2rad(lat1)) * cos(deg2rad(lat2)) * cos(deg2rad(theta));
         dist = acos(dist);
         dist = rad2deg(dist);
         dist = dist * 60 * 1.1515;
          if (unit == 'K') {
            dist = dist * 1.609344;
          } else if (unit == 'N') {
            dist = dist * 0.8684;
          }
          return dist.toStringAsFixed(2);
        }

    double deg2rad(double deg) {
      return (deg * pi / 180.0);
    }

    double rad2deg(double rad) {
      return (rad * 180.0 / pi);
    }

    void main() {
      print("Distance : " + distance(32.9697, -96.80322, 29.46786, -98.53506, 'K'));
    }
ocebsuys

ocebsuys5#

我使用vector_math将度数转换为弧度,还使用geolocator获取当前用户的纬度和经度,如果从当前位置搜索的话,还有一种方法可以直接计算两个位置之间的距离Geolocator.distanceBetween(startLatitude,startLongitude,endLatitude,endLongitude)

import 'dart:math' show sin, cos, sqrt, atan2;
import 'package:vector_math/vector_math.dart';
import 'package:geolocator/geolocator.dart';

Position _currentPosition;
double earthRadius = 6371000; 

//Using pLat and pLng as dummy location
double pLat = 22.8965265;   double pLng = 76.2545445; 

//Use Geolocator to find the current location(latitude & longitude)
getUserLocation() async {
   _currentPosition = await GeoLocator.getCurrentPosition(desiredAccuracy: LocationAccuracy.high);
}

//Calculating the distance between two points without Geolocator plugin
getDistance(){
   var dLat = radians(pLat - _currentPosition.latitude);
   var dLng = radians(pLng - _currentPosition.longitude);
   var a = sin(dLat/2) * sin(dLat/2) + cos(radians(_currentPosition.latitude)) 
           * cos(radians(pLat)) * sin(dLng/2) * sin(dLng/2);
   var c = 2 * atan2(sqrt(a), sqrt(1-a));
   var d = earthRadius * c;
   print(d); //d is the distance in meters
}

//Calculating the distance between two points with Geolocator plugin
getDistance(){
   final double distance = Geolocator.distanceBetween(pLat, pLng, 
           _currentPosition.latitude, _currentPosition.longitude); 
   print(distance);
}
avwztpqn

avwztpqn6#

  1. Geolocator.distanceBetween(lat1,lon1,lat2,lon2);
  2. print(“距离:${(distance / 1000).toStringAsFixed(0)} KM”);
qrjkbowd

qrjkbowd7#

您可以使用此函数计算距离,并将latlon从和到目的地传递,它将返回所需的距离

import 'package:google_maps_flutter/google_maps_flutter.dart';import 'dart:math';

class DistanceService{
  static double findDistance(LatLng from, LatLng to){
    var lat1 = toRadian(from.latitude);
    var lng1 = toRadian(from.longitude);
    var lat2 = toRadian(to.latitude);
    var lng2 = toRadian(to.longitude);
    
    // Haversine Formula
    var dlong = lng2 - lng1;
    var dlat = lat2 - lat1;

    var res = pow(sin((dlat / 2)), 2) + cos(lat1) * cos(lat2) * pow(sin(dlong / 2), 2);
    res = 2 * asin(sqrt(res));
    double R = 6371;
    res = res * R;
    return res;
  }

  static double toRadian(double val){
    double one_deg = (pi) / 180;
    return (one_deg * val);
  }
}
kxxlusnw

kxxlusnw8#

add  geolocator: ^7.1.1 in pubyml

//get current lat and long

  var currentLocation = myLocation;
    dynamic coordinates =
        new Coordinates(myLocation.latitude, myLocation.longitude);

 //getting in meter distance

//Latitude2 and Longitude2 can get anywhere to measure the distance

double distanceInMeters = Geolocator.distanceBetween(Latitude1, Longitude1, Latitude2, Longitude2);

相关问题