我正在构建一个使用Core Data数据库将数据存储在产品上的应用程序。这些产品显示在UICollectionView
中。此集合视图中的每个单元格都显示其包含的产品的基本信息,包括图像。
虽然单元格相对较小,但它们显示的原始图像最好非常大,因为它们也应该能够以更大的图像视图显示。图像直接从我的CellForItemAtIndexPath:
方法
func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "Cell", for: indexPath) as! CollectionWineCell
var current: Product!
//product details are set on labels
//image is set
if current.value(forKey: "image") != nil {
let image = current.value(forKey: "image") as! WineImage
let loadedImage = UIImage(data: image.image)
cell.imageview.image = loadedImage
} else {
cell.imageview.image = UIImage(named: "ProductPlaceholder.png")
}
return cell
}
当产品系列增长时,滚动会变得更加崎岖,并且会丢弃很多帧。这对我来说很有意义,但到目前为止我还没有找到合适的解决方案。在线查看时,可以从URL(在线或文件路径)加载异步图像,但是从Core Data执行此操作似乎并不常见。
我已经尝试过使用异步获取请求:
let fetchRequest = NSFetchRequest<NSFetchRequestResult>(entityName:"ProductImage")
fetchRequest.predicate = NSPredicate(format: "product = %@", current)
let asyncRequest = NSAsynchronousFetchRequest(fetchRequest: fetchRequest) { results in
if let results = results.finalResult {
let result = results[0] as! ProductImage
let loadedImage = UIImage(data: result.image)
DispatchQueue.main.async(execute: {
cell.wineImage.image = loadedImage
})
}
}
_ = try! managedObjectContext.executeRequest(asyncRequest)
然而,这种方法似乎并不能解决问题
问题
当从Core Data显示大量数据(包括图像)时,如何在UICollectionView
中以不会导致滞后和帧丢失的方式加载图像?
答案 0 :(得分:1)
如果图像可以如你所说的那么大,那么更好的方法不是将它们保存在Core Data中,而是将它们放在文件中。将文件名存储在Core Data中,然后使用它来查找文件。
但这不是当前的问题。即使这样,你也会因为花时间打开和解码图像数据而减速。基本上,更好的方法是不这样做。在您的集合视图中,图像可能显示的尺寸远小于其完整尺寸。而不是使用完整大小的图像,生成更合适的缩略图,并在集合视图中使用它。无论是从下载还是从用户的照片库或任何地方获取图像,都可以生成缩略图。保留缩略图以在集合视图中使用。只在您真正需要时使用全尺寸图像。
网上有很多关于如何缩放图像的例子,所以我不会在这里包含它。