如果任务被取消,则不执行该块

Do not execute the block if task is cancelled

我有 MainViewController -> ThumbnailViewController -> ImageFullScreenViewController。顾名思义,我有一个主屏幕,从主屏幕转到显示图像集合的屏幕,选择图像后,我会全屏打开图像。

ThumbnailViewController中,我下载图片如下

private func getImages() {
        self.galleryImages.removeAll()
        for url in urls {
            let task = NSURLSession.sharedSession().dataTaskWithURL(url) { (data, response, error) in

                // errors are handled

                if let image = UIImage(data: data) {
                    self.galleryImages.append(image!)
                }
            }
            task.resume()
            requests.append(task)
        }
    }

viewDidLoad()我调用getImages()。在 ThumbnailViewControllerviewWillDisappear() 中,我取消了正在进行的任务。

override func viewWillDisappear(animated: Bool) {
    super.viewWillDisappear(animated)
    if isMovingFromParentViewController() {
        if requests.count > 0 {
            for request in requests {
                if request is NSURLSessionDataTask {
                    let task = request as! NSURLSessionDataTask
                    task.cancel()
                }
            }
        }
    }
}

问题是,当我打开 ThumbnailViewController 并立即返回时,如果我立即打开 ThumbnailViewController,在某些情况下我可以看到同一图像的两个副本(很少,但可以重现)。

经过调查,我发现取消 viewWillDisappear 中的 NSURLSessionDataTask 只会取消任务,但不会取消完成块(这是行为)。在极少数情况下,完成块会针对前一个 NSURLSessionDataTask 执行,从而结束与新 NSURLSessionDataTask 的响应的混合。

我该如何处理?

注意:galleryImages 是我在 ImageFullScreenViewController (UIPageViewController)

中重用的单例 属性

首先,您应该注意释放 ThumbnailViewController 对象。使用 weak self 声明捕获列表,另外,它会在 99% 的情况下消除您的问题(如果您在其他一些地方没有保留 self),尽管您的模型仍然不适合该单调

private func getImages() {
    self.galleryImages.removeAll()
    for url in urls {
        let task = NSURLSession.sharedSession().dataTaskWithURL(url) { [weak self] (data, response, error) in

            // errors are handled
            // replace self. with self?.

            if let image = UIImage(data: data) {
                self?.galleryImages.append(image!)
            }
        }
        task.resume()
        requests.append(task)
    }
}