Свифт || Возвращение класса, который может использоваться другими методами из вызова API - PullRequest
0 голосов
/ 16 апреля 2019

Я звоню в HERE Weather API из мобильного приложения и мне нужно вернуть текущую погоду в качестве объекта, чтобы другие методы могли использовать эту информацию в течение определенного периода времени (например, 30-минутные интервалы обновления).

У меня есть приблизительное представление об асинхронных вызовах, основанных на этом сайте https://fluffy.es/return-value-from-a-closure/, но я все еще сталкиваюсь с моей проблемой, когда я не могу получить доступ к объекту Погода, который я создаю за пределами закрытия Завершения завершения.

Я предоставил метод Get из моего класса Weather. Я также предоставил вызов этого метода Get в файле AppDelegates.

public static func Get(lat:String, long:String, completionHandler: @escaping (Weather?)  -> Void) {

    //Keys go here
    var request = URLRequest(
        url: URL(string: "https://weather.cit.api.here.com/weather/1.0/report.json?app_code=-fJW5To2WdHdwQyYr_9ExQ&app_id=2m83HBDDcwAqTC2TqdLR&product=observation&latitude="+lat+"&longitude="+long+"&oneobservation=true")!)
    request.httpMethod = "GET"


    URLSession.shared.dataTask(with: request, completionHandler: { dat, response, error in
        do {

            guard let json = try? JSONSerialization.jsonObject(with: dat as! Data, options: []) else {throw JSONParseError.missingJSON}

            //parse json for dictionaries
            guard let response = json as? [String: Any] else {throw JSONParseError.responseNotADictionary}
            guard let observations = response["observations"] as? [String: Any] else {throw JSONParseError.observationsNotADictionary}
            guard let locationJSON = observations["location"] as? [Any] else {throw JSONParseError.locationNotAnArray}
            guard let location = locationJSON.first as? [String: Any] else {throw JSONParseError.locationNotADictionary}
            guard let observationJSON = location["observation"] as? [Any] else {throw JSONParseError.observationNotAnArray}
            guard let observation = observationJSON.first as? [String: Any] else {throw JSONParseError.observationNotADictionary}

            //search dictionaries for values
            guard let feedCreation = response["feedCreation"] as? String else {throw JSONParseError.missingFeedCreationObject}
            guard let city = location["city"] as? String else {throw JSONParseError.missingCityObject}
            guard let state = location["state"] as? String else {throw JSONParseError.missingStateObject}
            guard let temperature = observation["temperature"] as? String else {throw JSONParseError.missingTemperatureObject}
            guard let icon = observation["icon"] as? String else {throw JSONParseError.missingIconObject}
            guard let iconName = observation["iconName"] as? String else {throw JSONParseError.missingIconNameObject}

            //create weather object and return
            guard let currentWeather = Weather(feedCreation: feedCreation,state: state,city: city,temperature: temperature,icon: icon,iconName: iconName) else {throw WeatherObjectCreationError.objectCreationFailed}

            completionHandler(currentWeather)

        } catch {
            print("JSON Serialization error")
        }

    }).resume()
}
Weather.Get(lat:"32.9005", long:"-96.7869", completionHandler: {currentWeather in
            if let testWeather = currentWeather{
//Works fine
print(testWeather.city)
                print("completionHandler")
            }
})
//Error says testWeather is not intialized 
print(testWeather.city)

В конце моего вызова Weather.Get я должен иметь доступ к объекту testWeather из других методов. В частности, метод, который изменяет ограничение скорости на основе текущей погоды в области, которая возвращается вызовом Weather.Get.

1 Ответ

1 голос
/ 16 апреля 2019

Чего вы не поняли, так это того, что обработчик завершения в вашем вызове GET также асинхронен, как и исходный вызов API погоды.Таким образом, вы не можете запустить любой код после этого, который зависит от того, что в это.Порядок событий следующий:

// 1
Weather.Get(lat:"32.9005", long:"-96.7869", completionHandler: {currentWeather in
    if let testWeather = currentWeather{ // 3!!!
        print(testWeather.city) // 4!!!!!!
        print("completionHandler")
    }
})
print(testWeather.city) // 2

Кроме того, что такое testWeather в последней строке?Это собственность, self.testWeather?Это должно быть.Но даже если это так, вы пренебрегли значением self.testWeather;if let testWeather является другим testWeather (он является локальным только для обработчика завершения и не может «вытекать» из него).Однако, даже если бы вы сделали это, он все равно не будет работать, потому что код все еще будет работать в неправильном порядке:

// 1
Weather.Get(lat:"32.9005", long:"-96.7869", completionHandler: {currentWeather in
    if let testWeather = currentWeather{ // 3
        print(testWeather.city) // 4
        print("completionHandler") 
        self.testWeather = testWeather // 5; better but it won't help the _next_ print
    }
})
print(testWeather.city) // 2

Тем не менее,запоминание записи в self.testWeather (5) по крайней мере позволит другому коду получить доступ к этому значению при условии, что он запускается позже .

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...