我有 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
) 中重用的单例属性
首先,您应该注意释放 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/
欢迎光临 OStack程序员社区-中国程序员成长平台 (https://ostack.cn/) | Powered by Discuz! X3.4 |