Цепочка издателей быстрого комбайна и получение каждого результата
В приведенном ниже примере я делаю сетевой запрос на загрузку фильмов разных жанров, а затем использую его для загрузки всех фильмов. Приемник возвращает только результаты фильма. Как я мог получать и жанры, и фильмы?
struct Genre: Codable, Identifiable{
let id: Int
let name: String
var movies: [Movie]?
}
struct Movie: Codable, Hashable, Identifiable {
let title: String
let id: Int
let posterPath: String?
let backdropPath : String?
var tagline: String?
}
loadGenres() is AnyPublisher<[Genre], Error>
fetchMoviesIn() is AnyPublisher<[Movie], Error>
class GenresViewModel: ObservableObject{
@Published var genres = [Genre]()
@Published var movies = [Movie]()
var requests = Set<AnyCancellable>()
init(){
NetworkManager.shared.loadGenres()
.flatMap{ genres in
genres.publisher.flatMap{ genre in
NetworkManager.shared.fetchMoviesIn(genre)
}
}
.collect()
.retry(1)
.receive(on: DispatchQueue.main)
.sink(receiveCompletion: { completion in
switch completion{
case .finished:
print("Finished loading all movies in every genre")
case .failure(let error):
print("Error: \(error)")
}
}, receiveValue: { [self] values in
let allMovies = values.joined()
self.movies = allMovies.map{$0}
})
.store(in: &self.requests)
}
}
Ответы
Зависит от того, как вы хотите собирать жанры и фильмы.
Например, вам нужен жанр и список фильмов в этом жанре? Результатом может быть массив файлов (Genre, [Movies])
.
NetworkManager.shared.loadGenres()
.flatMap { genres in
genres.publisher.setFailureType(to: Error.self)
}
.flatMap { genre in
NetworkManager.shared.fetchMoviesIn(genre)
.map { movies in (genre, movies) }
}
.collect()
Или, если вам нужен массив (Genre, Movie)
кортежей, то это аналогичный подход, но с дополнительным уровнем .flatMap
для получения отдельных фильмов.
NetworkManager.shared.loadGenres()
.flatMap { genres in
genres.publisher.setFailureType(to: Error.self)
}
.flatMap { genre in
NetworkManager.shared.fetchMoviesIn(genre)
.flatMap { movies in
movies.publisher.setFailureType(to: Error.self)
}
.map { movie in (genre, movie) }
}
.collect()
Чтобы ответить на ваш вопрос с комментарием, вы хотите вернуть обновленное Genre
, вы можете вернуть это вместо возврата кортежа. Имейте в виду, что, поскольку Genre
это структура, вам нужно будет создать переменную копию объекта ( genre
доступная в flatMap
замыкании - константа), обновить копию и вернуть ее:
NetworkManager.shared.loadGenres()
.flatMap { genres in
genres.publisher.setFailureType(to: Error.self)
}
.flatMap { genre in
NetworkManager.shared.fetchMoviesIn(genre)
.map { movies -> Genre in
var genreCopy = genre
genreCopy.movies = movies
return genreCopy
}
}
.collect()