URLSession приводит к нулевым данным

Я пытаюсь изучить Swift, и у меня есть небольшой проект с API мест Google.

У меня есть метод для получения информации о местах, который использует URLSession для быстрой отправки запроса:

func fetchRestaurantDetails(placeId: String) -> Void {
    let jsonURLString = "https://maps.googleapis.com/maps/api/place/details/json?placeid=\(placeId)&key=[MY API KEY]"
    guard let url = URL(string: jsonURLString) else { return}

    let urlRequest = URLRequest(url: url)

    // set up the session
    let config = URLSessionConfiguration.default
    let session = URLSession(configuration: config)


    _ = session.dataTask(with: urlRequest) { (data, response, error) in
        // check for any errors
        guard error == nil else {
            print("error calling GET on /todos/1")
            print(error!)
            return
        }
        // make sure we got data
        guard let responseData = data else {
            print("Error: did not receive data")
            return
        }
        // parse the result as JSON, since that's what the API provides
        do {
            let place = try JSONDecoder().decode(Result.self, from: responseData) // New in Swift 4, used to serialize json.  
            self.rest = place.result
        } catch  {
            print("error trying to convert data to JSON")
            return
        }
    }.resume()
}

Я использую этот метод для создания экземпляра типа Restaurants, который позже добавлю в список:

func createRestaurant(placeId: String) -> Restaurants {
    self.fetchRestaurantDetails(placeId: placeId)
    let rest = Restaurants(name: self.rest.name,
                           formatted_address: self.rest.formatted_address,
                           website: self.rest.website,
                           location: ((self.rest.geometry.location.lat,self.rest.geometry.location.lng)),
                           opening_hours: self.rest.opening_hours.weekday_text,
                           photo: restImg)
    return rest!
}

Но всякий раз, когда я возвращаюсь к «let rest = Restaurants (...)», все значения равны нулю. Когда я пытаюсь отладить его, он просто перескакивает через мои разделы «_ = session» вплоть до resume (), затем снова возвращается к сеансу и снова заканчивается на возобновить (). Нет данных. Я очень озадачен, так как раньше я успешно выполнил этот фрагмент кода, и теперь мне интересно, не упустил ли я что-то. Спасибо :-)


person em0605    schedule 24.07.2017    source источник


Ответы (2)


Поставьте две точки останова. Один в

    let place = try JSONDecoder().decode(Result.self, from: responseData) // New in Swift 4, used to serialize json.  
    self.rest = place.result

а второй на

let rest = Restaurants(name: self.rest.name,
                       formatted_address: self.rest.formatted_address,
                       website: self.rest.website,
                       location: ((self.rest.geometry.location.lat,self.rest.geometry.location.lng)),
                       opening_hours: self.rest.opening_hours.weekday_text,
                       photo: restImg)

Вы поймете, что первым вызывается второй. Вы получаете данные, которые выполняются асинхронно, и до того, как они станут доступны, вы пытаетесь их использовать. Вы должны убедиться, что данные доступны, прежде чем использовать их. Один из способов здесь - использовать обработчик завершения. Вы можете узнать об обработчиках завершения здесь.

person Mohammad Sadiq    schedule 24.07.2017
comment
Спасибо! Это тоже было мое собственное первоначальное ощущение, но, думаю, я просто не мог осмыслить обработчики завершения! Спасибо. - person em0605; 25.07.2017

fetchRestaurantDetails является асинхронным методом из-за того, что вы вызываете в нем session.dataTask, который является асинхронным.

Вы пытаетесь использовать результаты функции до того, как она действительно вернется. У вас есть несколько способов решить эту проблему:

  1. Используйте обработчик завершения, чтобы вернуть значение из fetchRestaurantDetails
  2. Используйте DispatchGroups, чтобы определить, когда URLRequest закончился
  3. Используйте стороннюю платформу, такую ​​как PromiseKit, для обработки асинхронных функций, таких как обычные функции с возвращаемыми значениями.
person Dávid Pásztor    schedule 24.07.2017