问题描述
我最近遇到了两个数据获取(下载)API,它们对我来说似乎执行相同的操作。我看不到何时应该使用另一个。
我可以使用URLSession.shared.dataTask
var tasks: [URLSessionDataTask] = []
func loadItems(tuple : (name : String,imageURL : URL)) {
let task = URLSession.shared.dataTask(with: tuple.imageURL,completionHandler :
{ data,response,error in
guard let data = data,error == nil else { return }
dispatchQueue.main.async() { [weak self] in
self?.displayFlag(data: data,title: tuple.name)
}
})
tasks.append(task)
task.resume()
}
deinit {
tasks.forEach {
$0.cancel()
}
}
或者我可以使用URLSession.shared.dataTaskPublisher
var cancellables: [AnyCancellable] = []
func loadItems(tuple : (name : String,imageURL : URL)) {
URLSession.shared.dataTaskPublisher(for: tuple.imageURL)
.sink(
receiveCompletion: {
completion in
switch completion {
case .finished:
break
case .failure( _):
return
}},receiveValue: { data,_ in dispatchQueue.main.async { [weak self] in self?.displayFlag(data: data,title: tuple.name) } })
.store(in: &cancellables)
}
deinit {
cancellables.forEach {
$0.cancel()
}
}
我看不到它们的明显区别,因为两者都可以获取,并且都为我们提供了轻松取消任务的能力。有人可以阐明他们在何时使用哪种方面的差异吗?
解决方法
第一个是经典。它已经存在了很长一段时间,并且即使不是所有的开发人员也都对它熟悉。
第二个是第一个的包装,并允许将其与其他发布者合并(例如,仅在执行前两个请求时才执行某些请求)。使用第一种方法组合数据任务将更加困难。
因此,要点:将第一个请求用于一次性请求。当需要更多逻辑以将结果与其他发布者组合/传递给其他发布者(不仅来自URLSession)时,请使用第二个。基本上,这就是Combine框架背后的思想-您可以组合不同方式的异步机制(利用回调的数据任务就是其中之一)。
有关更多信息,请参见去年的WWDC视频中的介绍合并。