1

我正在使用 UICollectionView 在我的应用程序中显示图像。
问题是显示图像的速度非常慢。50 秒后,将出现集合视图中的图像。:(
当我在谷歌找到解决方案时,他们大多编写以下代码。
但这对我不起作用

  cell.layer.shouldRasterize = true
  cell.layer.rasterizationScale = UIScreen.main.scale

extension SeeAllCollectionView {

override func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
    debugPrint("seeAllLIStCell Count \(assetsTable.count)")
    return assetsTable.count
}

override func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {


    let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "seeAllListCell", for: indexPath) as! SeeAllPhotoCell

    let list = assetsTable[(indexPath as NSIndexPath).row]


    var imageName: String? = (list.poster_image_url)
    var image: UIImage? = (images_cache[imageName!])
    if image != nil {
        debugPrint("Yes Image")

        cell.imageView.image = image

    } else{
        debugPrint("NO Image")
        cell.imageView.image = nil

        DispatchQueue.main.async(){
            let url = NSURL(string: list.poster_image_url)
            let data = NSData(contentsOf:url! as URL)
            var image = UIImage(data: data as! Data)
            DispatchQueue.main.async(execute: {() -> Void in
                cell.movieTitle.text = list.name
                cell.imageView?.image = image
            })
            self.images_cache[imageName!] = image
        }

    }




    return cell
}

}


// MARK: - UICollectionViewDelegate

extension SeeAllCollectionView {

override func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {

            debugPrint("Selected")


    let list = assetsTable[(indexPath as NSIndexPath).row]
    debugPrint(list.poster_image_url)
    debugPrint(list.name)

    prefs.set(list.poster_image_url, forKey: "poster_image_url")
    prefs.set(list.name, forKey: "name")
    prefs.set(list.assets_id, forKey: "VIDEO_ID")
    prefs.set(false, forKey: "FLAG")
            let storyboard = UIStoryboard(name: "Main", bundle: nil)
            let vc = storyboard.instantiateViewController(withIdentifier: "DetailsChannel") as UIViewController
            self.present(vc, animated: true, completion: nil)

}

}

这是我运行项目时的屏幕截图。当我运行时,我得到了这么多行代码。

在此处输入图像描述

请任何人帮助我该怎么办?

4

3 回答 3

1

console error当我从 API 调用获取数据并重新加载UITableView(根据我的要求)时,我得到了同样的结果。我的问题通过使用解决了

DispatchQueue.global(qos: .background).async { // load data in back ground mode so that main thread can be safed.
let url = NSURL(string: list.poster_image_url)
            let data = NSData(contentsOf:url! as URL)
            var image = UIImage(data: data as! Data)
    DispatchQueue.main.async(execute: {
                                cell.movieTitle.text = list.name
                cell.imageView?.image = image
                            }) 
self.images_cache[imageName!] = image       
        }

我之前在控制台上遇到的错误的屏幕截图

在此处输入图像描述

于 2017-03-29T08:39:10.547 回答
0

这一行:

DispatchQueue.main.async()

下载图像时可能会导致错误,它会阻塞主线程并在主(UI)队列上执行网络请求,更不用说这些请求随后是串行执行的(因此很慢)。尝试将代码段更改为:

DispatchQueue.global.async(qos: DispatchQoS.QoSClass.background){
        let url = NSURL(string: list.poster_image_url)
        let data = NSData(contentsOf:url! as URL)
        var image = UIImage(data: data as! Data)
        DispatchQueue.main.async(execute: {() -> Void in

whereglobal代表此类网络操作的全局队列默认值。

编辑:除了使用主队列进行网络调用之外,实际上可能存在一个问题,即为当前在屏幕上不可见的行加载了太多图像。如果它们很多并且连接不太好,您最终将获得屏幕单元的应用程序待下载。考虑仅对屏幕单元格延迟加载图像 - 并取消不可见行的下载。在 Swift 中有一个非常好的教程(但是对于表格视图,但您可以轻松地将其扩展为收藏)关于如何完成此操作。

于 2017-03-29T08:49:48.560 回答
0

您可以尝试将SDWebimage的所有异步线程操作都维护得很好。

这可以提供以下优点

异步下载

如果应用程序发生内存警告,则自动清除图像缓存

图片 URL 缓存

图像缓存

避免重复下载

您可以直接在单元格中使用单个方法,如下所示

[cell.storeImg sd_setImageWithURL:[NSURL URLWithString:strURL] placeholderImage:kDefaultImageForDisplay];

在您的代码中,以下行应该会产生问题,因为该操作可能发生在主线程上 let data = NSData(contentsOf:url! as URL)

于 2017-03-29T09:21:03.950 回答