如果任务被取消,则不执行该块
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()
。在 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
只会取消任务,但不会取消完成块(这是行为)。在极少数情况下,完成块会针对前一个 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)
}
}
我有 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
只会取消任务,但不会取消完成块(这是行为)。在极少数情况下,完成块会针对前一个 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)
}
}