4

我有一个应用程序,我可以像 Uber 应用程序一样计算行驶距离。当驾驶员开始旅行时,即使在搜索行程中指定了起点,位置也开始发生变化,驾驶员可能会决定通过替代路线或通过较长的地方和路线,因为他/她不知道最短路线,那么我如何计算总距离。

起始位置是驾驶员按下开始按钮的位置 结束位置是驾驶员按下停止按钮的位置

这是我到目前为止的代码

    public func locationManager(_ manager: CLLocationManager, didUpdateLocations locations: [CLLocation]) {
        lastLocation = locations.last!
        endTrip(locations.last)

        if !hasSetInitialLocation {

            let camera = GMSCameraPosition.camera(withTarget: lastLocation!.coordinate, zoom: 17)
            self.mapView.animate(to: camera)
            hasSetInitialLocation = true
            endTrip(lastLocation)
            MqttManager.instance.connectToServer()
        }
    }



func endTrip(endLoaction: CLLocation) {
        guard let statusChange = source.getStatusChange() else{return}
        var distanceTraveled: Double = 0.0
        let initialLocation = CLLocation(latitude: (statusChange.meta?.location?.lat)!, longitude: (statusChange.meta?.location?.lng)!)
        let distance = initialLocation.distance(from: endLoaction)
        distanceTraveled += distance
        let distanceInKM = Utility.convertCLLocationDistanceToKiloMeters(targetDistance: distanceTraveled)
}

我如何计算距离以反映驾驶员移动的总距离,因为从建议的起点和终点的路线可能会发生变化。

司机按下了一个叫做 start trip 的按钮,我想得到从那一刻到他按下按钮 end trip 的距离

这个实现可以从类似的工作代码中获得,但唯一的区别是它们是一个开始按钮,它传递该点的坐标和一个停止坐标,它是坐标的结束。

enum DistanceValue: Int {
                case meters, miles
            }

            func calculateDistanceBetweenLocations(_ firstLocation: CLLocation, secondLocation: CLLocation, valueType: DistanceValue) -> Double {
                var distance = 0.0
                let meters = firstLocation.distance(from: secondLocation)
                distance += meters
                switch valueType {
                case .meters:
                    return distance
                case .miles:
                    let miles = distance
                    return miles
                }
            }

    func locationManager(_ manager: CLLocationManager, didUpdateLocations locations: [CLLocation]) {
            if startLocation == nil {
                startLocation = locations.first
            } else if let location = locations.last {
                runDistance += lastLocation.distance(from: location)
                let calc = calculateDistanceBetweenLocations(lastLocation, secondLocation: location, valueType: .meters)

                print("TOTAL LOC 1 \(calc)")
                print("TOTAL LOC 2 \(runDistance)")
            }
            lastLocation = locations.last

        }

如我的打印声明所示,我print("TOTAL LOC 1 \(calc)") print("TOTAL LOC 2 \(runDistance)")该怎么做

calcrunDistance

这是控制台中打印的内容

TOTAL LOC 10.29331530774379
TOTAL LOC 2 10.29331530774379
TOTAL LOC 2.2655118031831587
TOTAL LOC 2 12.558827110926948
4

3 回答 3

4

如果您使用第一个和最后一个坐标获得这样的距离,它总是返回错误的值,因为它无法识别实际的行进路径。

我确实使用以下代码解决了同样的问题。

使用谷歌地图

> pod 'GoogleMaps'

当驾驶员在路线上移动时制作坐标数组。

var arr = [Any]() 
// Driving lat long co-ordinateds continues add in this array according to your expectation either update location or perticuler time duration.

// make GMSMutablePath of your co-ordinates
let path = GMSMutablePath()

    for obj in arr{

        print(obj)

        if let lat = (obj as? NSDictionary)?.value(forKey: PARAMETERS.LET) as? String{

            path.addLatitude(Double(lat)!, longitude: Double(((obj as? NSDictionary)?.value(forKey: PARAMETERS.LONG) as? String)!)!)

        }
    }

print(path) // Here is your traveling path
let km = GMSGeometryLength(path)
print(km) // your total traveling distance.

我在这个应用程序中做到了,它工作正常。 希望它会帮助你:)

或者没有谷歌地图

不过,根据您的代码,您必须为自己提供位置,一个 CLLocationCoordinate2D 数组。

class ViewController: UIViewController, CLLocationManagerDelegate, MKMapViewDelegate {
    // MARK: - Variables
    let locationManager = CLLocationManager()

    // MARK: - IBOutlet
    @IBOutlet weak var mapView: MKMapView!

    // MARK: - IBAction
    @IBAction func distanceTapped(_ sender: UIBarButtonItem) {
        let locations: [CLLocationCoordinate2D] = [...]
        var total: Double = 0.0
        for i in 0..<locations.count - 1 {
            let start = locations[i]
            let end = locations[i + 1]
            let distance = getDistance(from: start, to: end)
            total += distance
        }
        print(total)
    }

    func getDistance(from: CLLocationCoordinate2D, to: CLLocationCoordinate2D) -> CLLocationDistance {
        // By Aviel Gross
        // https://stackoverflow.com/questions/11077425/finding-distance-between-cllocationcoordinate2d-points
        let from = CLLocation(latitude: from.latitude, longitude: from.longitude)
        let to = CLLocation(latitude: to.latitude, longitude: to.longitude)
        return from.distance(from: to)
    }
}

输出

在此处输入图像描述

于 2019-03-12T07:22:08.343 回答
0

给定一个 CLLocationCoordinate2D 数组,计算距离(以米为单位)的简单函数。使用reduce而不是数组迭代。

func computeDistance(from points: [CLLocationCoordinate2D]) -> Double {
    guard let first = points.first else { return 0.0 }
    var prevPoint = first
    return points.reduce(0.0) { (count, point) -> Double in
        let newCount = count + CLLocation(latitude: prevPoint.latitude, longitude: prevPoint.longitude).distance(
            from: CLLocation(latitude: point.latitude, longitude: point.longitude))
        prevPoint = point
        return newCount
    }
}
于 2020-04-15T22:16:21.713 回答
0

我喜欢为此使用扩展名

extension Array where Element: CLLocation {
    
    var distance: Double {
        guard count > 1 else { return 0 }
        var previous = self[0]
        
        return reduce(0) { (result, location) -> Double in
            let distance = location.distance(from: previous)
            previous = location
            return result + distance
        }
    }
    
}

用法:

locations.distance
于 2022-02-15T21:13:50.640 回答