如何从CLGeocoder返回值?

时间:2018-04-03 09:23:02

标签: ios swift core-location

我正在开发天气应用程序作为培训,需要将位置转换为城市。所以我正在使用CLGeocoder:

 func updateWeatherData(json: JSON?) {

        if let json = json {
            weatherData.temperature = fahrenheitToCelcius(json["currently"]["temperature"].doubleValue)
            weatherData.weatherIconName = json["currently"]["icon"].stringValue

            let location = CLLocation(latitude: json["latitude"].doubleValue, longitude: json["longitude"].doubleValue)
            CLGeocoder().reverseGeocodeLocation(location) { (placemark, error) in

                if let city = placemark {
                    self.weatherData.city = city.last?.locality
                } else if let error = error {
                    print(error)
                }
            }
            updateUIWithWeatherData()
        }

    }
    //MARK: - UI Updates

    func updateUIWithWeatherData() {

        cityLabel.text = weatherData.city
        temperatureLabel.text = "\(weatherData.temperature)°"
        weatherIcon.image = UIImage(named: weatherData.weatherIconName!)

    }

此代码将nil返回给weatherData.city。但是当我在一个闭包中放置一个断点时,一切正常。我错过了什么?

1 个答案:

答案 0 :(得分:0)

如果我正确理解您的问题,您需要在地理编码完成后更新您的用户界面。如下所示:

func updateWeatherData(json: JSON?) {

    if let json = json {
        weatherData.temperature = fahrenheitToCelcius(json["currently"]["temperature"].doubleValue)
        weatherData.weatherIconName = json["currently"]["icon"].stringValue

        let location = CLLocation(latitude: json["latitude"].doubleValue, longitude: json["longitude"].doubleValue)
        CLGeocoder().reverseGeocodeLocation(location) { (placemark, error) in

            if let city = placemark {
                self.weatherData.city = city.last?.locality
            } else if let error = error {
                print(error)
            }
            self.updateUIWithWeatherData()
        }            
    }

}

这些操作的顺序是,地理编码是异步完成的,并且可能在稍后调用的代码之后发生。注意可能但不需要。

您还应该阅读有关线程的此方法的文档。 UI必须在主线程上更新,所以除非文档指定调用将在主线程上完成,否则最好强制它:

CLGeocoder().reverseGeocodeLocation(location) { (placemark, error) in    
    if let city = placemark {
        self.weatherData.city = city.last?.locality
    } else if let error = error {
        print(error)
    }
    DispatchQueue.main.async {
        self.updateUIWithWeatherData()
    }
}