gpt4 book ai didi

ios - Swift:使用 NSOperation 保留循环

转载 作者:搜寻专家 更新时间:2023-10-31 19:38:27 26 4
gpt4 key购买 nike

在我的应用程序中,我使用图像加载器类从 Web 加载图像以用于 Collection View 。该类跟踪下载操作并在图像的单元格在 Collection View 中不再可见时取消它们。此实现基于 NSOperation 的 raywenderlich 教程:http://www.raywenderlich.com/76341/use-nsoperation-nsoperationqueue-swift .

我使用 NSOperation 从网络上下载图像。我注意到 Instruments 没有发布任何 NSoperations。这会增加下载的每个图像的已用内存。在完成 block 中,我引用了“self”。所以我发现我创建了一个保留周期。

我在网上看了很多例子。我知道我可以将捕获列表与“弱 self ”或“无主 self ”一起使用。我在完成 block 中尝试了这个,但仍然没有释放操作。

我的图片加载类代码如下:

import Foundation
import UIKit

class ImageLoader {
lazy var downloadsInProgress = [NSIndexPath:NSOperation]()
lazy var downloadQueue:NSOperationQueue = {
var queue = NSOperationQueue()
queue.name = "Image Download queue"
return queue
}()

let cache = NSCache() // contains NSData objects for images

init() {
// Max. cache size is 10% of available physical memory (in MB's)
cache.totalCostLimit = 200 * 1024 * 1024 // TODO: change to 10%
}

/**
* Download image based on url for given indexpath.
* The download is only started if the indexpath is still present in the downloadsInProgress array
*/

func startDownloadForUrl(url: String, indexPath: NSIndexPath, completion: (imageData: NSData?) -> Void) {
// check if download request is already present
if downloadsInProgress[indexPath] != nil {
return
}

// check cache
if let imageData = self.cache.objectForKey(url) as? NSData {
NSOperationQueue.mainQueue().addOperationWithBlock() {
//remove indexpath from progress queue
self.downloadsInProgress.removeValueForKey(indexPath)
completion(imageData: imageData)
}
return
}

// prepare the download
let downloader = ImageDownloader(url: url)

downloader.completionBlock = {
[unowned self] in

if downloader.cancelled {
return
}

// image is retrieved from web
NSOperationQueue.mainQueue().addOperationWithBlock() {
[unowned self] in

//remove indexpath from progress queue
self.downloadsInProgress.removeValueForKey(indexPath)

// add image to cache
if downloader.imageData != nil {
self.cache.setObject(downloader.imageData!, forKey: url, cost: downloader.imageData!.length)
}
completion(imageData: downloader.imageData)
}
}

// add downloader to operations in progress and start the operation
NSOperationQueue.mainQueue().addOperationWithBlock() {
[unowned self] in

self.downloadsInProgress[indexPath] = downloader
self.downloadQueue.addOperation(downloader)
}
}


/**
* Suspends queue for downloading images
*/

func suspendAllOperations() {
downloadQueue.suspended = true
}


/**
* Resumes queue for downloading images
*/

func resumeAllOperations() {
downloadQueue.suspended = false
}


/**
* Cancels downloads for NOT visible indexpaths. The parameter specifies an array of visible indexpaths!
*/

func cancelDownloads(visibleIndexPaths: [NSIndexPath]) {
let allPendingOperations = Set(downloadsInProgress.keys)
let visiblePaths = Set(visibleIndexPaths)

// cancel all pending operations for indexpaths that are not visible
var toBeCancelled = allPendingOperations
toBeCancelled.subtractInPlace(visiblePaths)

for indexPath in toBeCancelled {
if let pendingDownloadOperation = downloadsInProgress[indexPath] {
pendingDownloadOperation.cancel()
}

downloadsInProgress.removeValueForKey(indexPath)
}
}
}


class ImageDownloader: NSOperation {
var url: String
var imageData: NSData?

init(url: String) {
self.url = url
}

override func main() {
if self.cancelled {
return
}

if let imageUrl = NSURL(string: url) {
// retrieve data from web
setNetworkActivityIndicatorVisible(true)
imageData = NSData(contentsOfURL: imageUrl)
setNetworkActivityIndicatorVisible(false)

if self.cancelled {
imageData = nil
return
}

// scale image
if imageData != nil {
if let image = UIImage(data: imageData!) {
let imageData2 = UIImageJPEGRepresentation(image, 1.0)
let compressionRate = Float(imageData!.length) / Float(imageData2!.length)

let scaleWidth = 244 / image.size.width
let scaleHeight = 244 / image.size.height
let imageScale = min(scaleWidth, scaleHeight)

let rect = CGRectMake(0.0, 0.0, image.size.width * imageScale, image.size.height * imageScale)

UIGraphicsBeginImageContext(rect.size)
image.drawInRect(rect)
let scaledImage = UIGraphicsGetImageFromCurrentImageContext()
let scaledImageData = UIImageJPEGRepresentation(scaledImage, CGFloat(compressionRate))
UIGraphicsEndImageContext()

imageData = scaledImageData
}
}
}
}

private func setNetworkActivityIndicatorVisible(visible: Bool) {
NSOperationQueue.mainQueue().addOperationWithBlock() {
let appDelegate = UIApplication.sharedApplication().delegate as! AppDelegate
appDelegate.setNetworkActivityIndicatorVisible(visible)
}
}
}

我究竟在哪里创建保留周期?我该如何解决这个问题?什么时候应该使用“无主”,什么时候应该使用“弱”?

如果有人能解释解决方案,我将不胜感激,这样我就可以从错误中吸取教训。

最佳答案

我发现了问题。 retain cycle不是引用self引起的,而是在NSOperation的completion block中引用了NSOperation!

在函数 startDownloadForUrl(...) 中,我声明了变量 downloader。接下来,我为此变量声明一个完成 block 。在这个完成 block 中,我引用了变量 downloader。这会导致保留周期。

我通过在完成 block 中使用 [unowned downloader] 解决了这个问题。

这又产生了另一个问题。在完成 block 中,我异步调用主线程。在此调用中,使用了变量 downloader.imageData。由于这个异步调用,NSOperation 可能已经结束并且变量 downloader 可能不再存在。为避免崩溃,我为 imageData 声明了一个新变量,因此数据在主线程中使用时仍然可用。

完成 block 现在看起来像:

downloader.completionBlock = {
[unowned downloader] in
if downloader.cancelled {
return
}

let imageData = downloader.imageData // retain the imageData. It will be used asynchrounous in the main thread. The downloader operation might already be finished and downloader will no longer exists.

// image is retrieved from web
NSOperationQueue.mainQueue().addOperationWithBlock() {
//remove indexpath from progress queue
self.downloadsInProgress.removeValueForKey(indexPath)

// add image to cache
if imageData != nil {
self.cache.setObject(imageData!, forKey: url, cost: imageData!.length)
}
completion(imageData: imageData)
}
}

关于ios - Swift:使用 NSOperation 保留循环,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34528333/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com