URLSession.shared.dataTask与dataTaskPublisher,什么时候使用?

问题描述

我最近遇到了两个数据获取(下载)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视频中的介绍合并。