从CloudKit获取CKAsset图像非常慢

时间:2016-02-17 05:01:27

标签: ios swift cloudkit

我使用CloudKit作为iOS应用程序的服务器后端。我用它来存放一些相对静态的数据以及一些图像(CKAsset)。当我花时间从公共数据库中实际获取这些资产时,我遇到了一个问题。它们以极慢的速度加载。

我的用例是将图像加载到集合视图中的每个单元格中。图像的大小仅为200kb,但获取过程平均需要2.2秒才能完成下载并在单元格中设置图像。为了比较,我使用类似大小的库存图像的URL并使用NSURLSession加载它们。每个图像加载仅需0.18 - 0.25秒。

我尝试过多种不同的方式从CK下载图像:直接获取记录,查询和操作查询。所有这些都有类似的结果。在为单元格设置图像之前,我还将调度回完成块中的主队列。

我的数据库设置为具有包含多个数据字段的主对象。然后我为照片设置了一个向后的参考样式系统,其中每张照片只有一个主要对象的引用。这样我就可以按需加载照片,而不会影响主要数据。

它看起来像这样:

主要对象: title: String, startDate: Date

照片对象: owner: String(reference to primary object), image: Asset

以下是我尝试直接获取其中一张照片的示例请求:

let publicDb = CKContainer.defaultContainer().publicCloudDatabase
let configRecordId = CKRecordID(recordName: "e783f542-ec0f-46j4-9e99-b3e3ez505adf")

publicDb.fetchRecordWithID(configRecordId) { (record, error) -> Void in
    dispatch_async(dispatch_get_main_queue()) {
        guard let photoRecord = record else { return }
        guard let asset = photoRecord["image"] as? CKAsset else { return }

        guard let photo = NSData(contentsOfURL: asset.fileURL) else { return }

        let image = UIImage(data: photo)!

        cell.cardImageView.image = image
    }
}

我似乎无法弄清楚为什么这些图像下载花了这么长时间,但如果我无法让它们在合理的时间内加载,那真的是非常明显的。

更新:我尝试使用较小的图像23kb进行获取操作。获取速度更快,从0.3到1.1秒不等。那更好,但仍然不符合我对CloudKit应该提供的期望。

2 个答案:

答案 0 :(得分:7)

我正在使用CKQueryOperation。我发现,一旦我将以下代码添加到我的代码中,下载CKAssets的速度提高了大约5-10倍。

    queryOperation.qualityOfService = .UserInteractive

这是我的完整代码:

func getReportPhotos(report:Report, completionHandler: (report:Report?, error:NSError?) -> ()) {
    let photo : Photo = report.photos![0] as! Photo
    let predicate : NSPredicate = NSPredicate(format: "recordID = %@", CKRecordID(recordName: photo.identifier!))
    let query : CKQuery = CKQuery(recordType: "Photo", predicate: predicate)
    let queryOperation : CKQueryOperation = CKQueryOperation()
    queryOperation.query = query
    queryOperation.resultsLimit = numberOfReportsPerQuery        
    queryOperation.qualityOfService = .UserInteractive
    queryOperation.recordFetchedBlock = { record in
        photo.date = record.objectForKey("date") as? NSDate
        photo.fileType = record.objectForKey("fileType") as? String
        let asset : CKAsset? = record.objectForKey("image") as? CKAsset
        if asset != nil {
            let photoData : NSData? = NSData(contentsOfURL:asset!.fileURL)
            let photo : Photo = report.photos![0] as! Photo
            photo.image = UIImage(data:photoData!)
        }

    }
    queryOperation.queryCompletionBlock = { queryCursor, error in
        dispatch_async(dispatch_get_main_queue(), {
            completionHandler(report: report, error: error)
        })
    }
    publicDatabase?.addOperation(queryOperation)
}

答案 1 :(得分:3)

主线程似乎有些慢,这会导致执行dispatch_async调用的捕获块的延迟。您的代码是否可能多次并行调用此记录获取函数?这将导致NSData(contentsOfURL:asset.fileURL)处理占用主线程并引入累积延迟。

在任何情况下,如果仅作为一种好的做法,使用NSData加载图像应该在后台执行,而不是在主线程上执行。