Результаты URLSession в данных NIL
Я пытаюсь изучить Swift, и у меня есть небольшой проект с API Google Мест.
У меня есть метод для получения сведений о местах, который использует URLSession в Swift для отправки запроса:
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 (...)", все значения равны нулю. Когда я пытаюсь его отладить, он просто перепрыгивает через мои разделы "_ = сеанс" вплоть до resume (), затем снова возвращается к сеансу и заканчивается в resume (). Данные не получены. Я довольно озадачен, так как я успешно выполнил этот кусок кода раньше, и теперь мне интересно, если я что-то пропустил. Спасибо:-)
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)
Вы поймете, что второго называют первым. Вы извлекаете данные, что делается асинхронно, и до того, как они становятся доступными, вы пытаетесь их использовать. Вы должны убедиться, что данные доступны, прежде чем использовать их. Один из способов - использовать обработчик завершения. Вы можете узнать о обработчиках завершения здесь.
fetchRestaurantDetails
это асинхронный метод из-за того, что вы вызываете session.dataTask
в нем, который является асинхронным.
Вы пытаетесь использовать результаты функции до ее фактического возврата. У вас есть несколько способов решить эту проблему:
- Используйте обработчик завершения, чтобы вернуть значение из
fetchRestaurantDetails
- использование
DispatchGroups
обнаружить, когдаURLRequest
законченный - Используйте стороннюю платформу, такую как PromiseKit, для обработки асинхронных функций, таких как обычные функции, с возвращаемыми значениями.