【发布时间】:2020-11-16 08:33:27
【问题描述】:
我一直在寻找从远程服务器图像 URL 异步加载图像的良好解决方案。网上有很多解决方案。遗憾的是,Apple 没有为如此常见的东西提供原生服务。不管怎样,我发现Sundell's blog 真的很有趣,并从中汲取了一些好处,创建了我自己的 ImageLoader,如下所示:
import Combine
class ImageLoader {
private let urlSession: URLSession
private let cache: NSCache<NSURL, UIImage>
init(urlSession: URLSession = .shared,
cache: NSCache<NSURL, UIImage> = .init()) {
self.urlSession = urlSession
self.cache = cache
}
func publisher(for url: URL) -> AnyPublisher<UIImage, Error> {
if let image = cache.object(forKey: url as NSURL) {
return Just(image)
.setFailureType(to: Error.self)
.receive(on: DispatchQueue.main)
.eraseToAnyPublisher()
} else {
return urlSession
.dataTaskPublisher(for: url)
.map(\.data)
.tryMap { data in
guard let image = UIImage(data: data) else {
throw URLError(.badServerResponse, userInfo: [
NSURLErrorFailingURLErrorKey: url
])
}
return image
}
.receive(on: DispatchQueue.main)
.handleEvents(receiveOutput: { [cache] image in
cache.setObject(image, forKey: url as NSURL)
})
.eraseToAnyPublisher()
}
}
}
如您所见,发布者提供了AnyPublisher<UIImage, Error> 的实例。我不完全确定如何在我的MyImageView 中使用这个ImageLoader,如下所示:
struct MyImageView: View {
var url: URL
var imageLoader = ImageLoader()
@State private var image = #imageLiteral(resourceName: "placeholder")
var body: some View {
Image(uiImage: image)
.onAppear {
let cancellable = imageLoader.publisher(for: url).sink(receiveCompletion: { failure in
print(failure) // doesn't print
}, receiveValue: { image in
self.image = image // not getting executed
})
cancellable.cancel() // tried with and without this line.
}
}
}
如何从返回AnyPublisher<UIImage, Error> 实例的ImageLoader 发布者中提取UIImage?
【问题讨论】:
-
您的
let cancellable将在您完成执行onAppear后立即释放。您需要将引用保留足够长的时间以执行请求。最好将其移至某个 class(如ObservableObject)。请注意,您仍然需要在类级别声明您的cancellable,因此它不会立即被释放。 -
@pawello2222 如果我让
ImageLoader符合ObservableObject会不会出错? -
不,如果您不想要太多类,将
ImageLoader与ObservableObject一致是有意义的。 -
@pawello2222 我在下面添加了我的方法,让我知道它需要任何更改或可以进一步改进。
标签: swift image swiftui combine publisher