我有 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() 。在 ThumbnailViewController 的 viewWillDisappear() 中,我取消了正在进行的任务。
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 只会取消任务,但不会取消完成 block (这是行为)。在极少数情况下,完成 block 会为先前的 NSURLSessionDataTask 执行,从而与新的 NSURLSessionDataTask 的响应混合结束。
我该如何处理?
注意:galleryImages 是我在 ImageFullScreenViewController (UIPageViewController ) 中重用的单例属性
Best Answer-推荐答案 strong>
首先,您应该注意释放 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)
}
}
关于ios - 如果任务被取消,则不执行该 block ,我们在Stack Overflow上找到一个类似的问题:
https://stackoverflow.com/questions/39527949/
|