2

为什么NSURLSession创建和恢复 NSURLSessionTask 后操作队列为空?

有没有办法判断 NSURLSession 是否有待处理的任务?

目标是等待多个任务完成,但这不起作用:

NSURLSessionUploadTask *uploadTask = [self.session uploadTaskWithStreamedRequest:request];
[uploadTask resume];
// this prints "0"
NSLog(self.session.delegateQueue.operationCount)
// this returns immediately instead of waiting for task to complete
[self.session.delegateQueue waitUntilAllOperationsAreFinished];
4

2 回答 2

10

我找到了一个避免使会话无效的解决方案,使用建议的DispatchGroup.

(答案在 Swift 中,而问题在 ObjC 中......但这是相同的逻辑)

请注意,在使用时uploadTaskWithStreamedRequest:,我们需要实现一个URLSessionTaskDelegateand func urlSession(_ session: URLSession, task: URLSessionTask, didCompleteWithError error: Error?)。因此,为了简化答案,我将演示DispatchGroupwith的用法uploadTaskWithRequest:from:completionHandler:

// strong reference to the dispatch group
let dispatchGroup = DispatchGroup()

func performManyThings() {
    for _ in 1...3 {
        let request = URLRequest(url: URL(string: "http://example.com")!)
        dispatchGroup.enter()
        let uploadTask = self.session.uploadTask(with: request, from: nil) { [weak self] _, _, _ in
            self?.dispatchGroup.leave()
        }
        uploadTask.resume()
    }
    dispatchGroup.notify(queue: .main) {
        // here, all the tasks are completed
    }
}
于 2016-12-13T12:04:44.080 回答
1

只要您不关心重用会话并且不介意异步执行操作,这非常容易:

  • 召集finishTasksAndInvalidate会议。
  • 在您的会话委托中实现该URLSession:didBecomeInvalidWithError:方法以在最后一个任务完成后执行您需要执行的任何工作。

也就是说,上面代码的问题是会话在开始第一次提取之前没有任何操作,只要您的代码阻塞运行循环,就不会发生这种情况。您通常不应该尝试同步使用 NSURLSession。解决问题的方法几乎总是错误的。:-)

于 2016-12-03T20:24:03.207 回答