CLLocationManager,核心数据和数组

Posted

技术标签:

【中文标题】CLLocationManager,核心数据和数组【英文标题】:CLLocationManager, Core Data and Arrays 【发布时间】:2016-12-29 19:02:13 【问题描述】:

我已经有了这个功能,而且离让它工作很近了!我想我遇到了转换问题。我可以将坐标转换为具有 3 个属性 lat(Double)、long(Double)和 time(Date)的实体。那行得通,我可以将内容打印到屏幕上,而且看起来不错。但是当我尝试提取并用作地图上的坐标时,它不起作用。我认为这是因为它们没有被识别为真正的坐标。有什么想法吗?

   func locationManager(_ manager: CLLocationManager, didUpdateLocations locations: [CLLocation])
    
        //ARRAY for lat/long data
        var latArr:[Double] = []
        var longArr:[Double] = []
        var timeArr:[Date] = []

        let location = locations[0]
        let annotation = MKPointAnnotation()
        let span:MKCoordinateSpan = MKCoordinateSpanMake(0.01, 0.01)
        let myLocation = CLLocationCoordinate2D(latitude: location.coordinate.latitude, longitude: location.coordinate.longitude)
        //annotation.coordinate = myLocation
        //annotation.title = "\(location.timestamp)"
        //self.mapView.addAnnotation(annotation)
        let region:MKCoordinateRegion = MKCoordinateRegionMake(myLocation, span)
        self.mapView.setRegion(region, animated: true)
        self.mapView.showsUserLocation = false //SET THIS
        latitude.text = "latitude: " + "\(location.coordinate.latitude)"
        longitude.text = "longitude: " + "\(location.coordinate.longitude)"
        altitude.text = "altitude: " + "\(location.altitude)"
        speed.text = "speed: " + "\(location.speed)"
        timestamp.text = "timestamp: " + "\(location.timestamp)"

        //START -- Core Data
        let appDelegate = UIApplication.shared.delegate as! AppDelegate
        let context = appDelegate.persistentContainer.viewContext

        let request = NSFetchRequest<NSFetchRequestResult>(entityName: "Data")
        request.returnsObjectsAsFaults = false

        let data = NSEntityDescription.insertNewObject(forEntityName: "Data", into: context)
        //let data = NSNumber(double: self.location?.coordinate.latitude)

        let dlat = Double("\(location.coordinate.latitude)")
        let dlong = Double("\(location.coordinate.longitude)")

        data.setValue(dlat, forKey: "lat")
        data.setValue(dlong, forKey: "long")
        data.setValue(location.timestamp, forKey: "time")
        //END -- Core Data

        do
        
            try context.save()
            //print("Saved")
        
        catch
        

        

        do
        
            var lat: Double = 0
            var long: Double = 0
            let t = NSDate()
            let results = try context.fetch(request)

            if results.count > 0
            
                for result in results as! [NSManagedObject]
                
                    if let datalat = result.value(forKey: "lat") as? Double
                    
                        //latArr.append(datalat)
                        let lat = datalat
                        print(lat)
                    
                    if let datalong = result.value(forKey: "long") as? Double
                    
                        //longArr.append(datalong)
                        let long = datalong
                        print(long)
                    
                    if let time = result.value(forKey: "time") as? Date
                    
                        //timeArr.append(time)
                        let t = time
                        print(t)
                    
                    let mLocation = CLLocationCoordinate2DMake(lat, long)
                    annotation.coordinate = mLocation
                    annotation.title = "\(t)"
                    self.mapView.addAnnotation(annotation)
                
            
        
        catch
        

        

【问题讨论】:

“它不起作用”具体是什么方式? 永远不会有空的 catch 语句。至少在其中放一个日志语句。错误可能会出现,但您现在永远不会知道它们,不是吗? 您确定显示的是地图的正确区域吗?为了安全起见,您可以添加 mapView.showAnnotation(annotation, animated: false) 【参考方案1】:

我将您的代码与我在类似场景中使用的函数进行了比较,并合并了任何差异(为了简洁起见,我做了一些编辑,您可以接受或离开)。相关部分如下:

    let request = NSFetchRequest<Data>(entityName: "Data")
let annotations: [MKAnnotation] = []

annotation.title = “\(t)”

for result in results  

    if data.lat != nil && data.lon != nil
    
        annotation.coordinate = CLLocationCoordinate2D(latitude: data.lat.doubleValue, longitude: data.lon.doubleValue)
        annotations.append(annotation)
    
    self.mapView.addAnnotations(annotations)
    self.mapView.showAnnotations(annotations, animated: false)

我的代码可以工作,但它是 Swift 3.0,因此如果您使用的是 2.2.或 2.3,可能会由于早期版本的更改而产生一些差异。我知道我需要将它从 NSNumber(来自 CoreData)推送到 Double,但我认为如果这是问题所在,你会得到一个编译错误 - 不知道你为什么不这样做。正如我在评论中提到的,如果航点只是在可见地图区域之外,则显示注释会有所帮助。

除此之外,您的代码在我看来还不错 - 如果它仍在运行,请告诉我。

【讨论】:

谢谢。我收到此错误:“NSManagedObject”类型的值没有成员“lat”。有任何想法吗?这个错误:“MKMapView”类型的值没有成员“showAnnotation” 我更改了获取请求以将其绑定到您的数据实体(顺便说一句,这不是一个理想的名称......)并意识到 showAnnotations 必须是复数 - 所以我将流程更改为使用数组,反正我的流程就是这样。 仍然挂断。 data.lat 来自哪里?另外,感谢您的所有帮助 不客气...对不起,从上面我的理解是,您的实体“数据”具有称为“纬度”和“经度”的字段,建议的代码对其进行测试然后使用计算CLLocation。发生了什么事?

以上是关于CLLocationManager,核心数据和数组的主要内容,如果未能解决你的问题,请参考以下文章

核心位置,是不是可以在没有 GPS 芯片的情况下向 iPad 提供 GPS/位置日期,以便在 CLLocationManager 类中使用?

CLLocationManager 不适用于非无线连接?

插值和预测 CLLocationManager

在 XCTest 案例中处理对 CLLocationManager 授权的请求

XCTest CLLocationManager 的委托方法不会被调用

CLLocationManager 和 tvOS - RequestWhenInUseAuthorization() 不提示