1

我正在尝试从 PHAsset 获取缩略图以及其他一些信息。这是一个简化的代码片段。

对于我从相机胶卷中检索到的所有图像,请执行以下操作:

static func fetchAsset(asset: PHAsset)
{
        if(asset.mediaType == PHAssetMediaType.Image){
            let screenScale: CGFloat = UIScreen.mainScreen().scale
            let imageSize = CGSize(width: 100 * screenScale, height: 100 * screenScale)

            let options: PHImageRequestOptions = PHImageRequestOptions()
            options.deliveryMode = PHImageRequestOptionsDeliveryMode.FastFormat
            options.resizeMode = PHImageRequestOptionsResizeMode.Fast
            options.synchronous = false

            PHImageManager.defaultManager().requestImageForAsset(asset,
                targetSize: imageSize,
                contentMode: PHImageContentMode.AspectFill,
                options: options,
                resultHandler: { (result, info) -> Void in
                    if (result != nil && result!.scale == screenScale) {

                        var newItemImage = MyImage(isLocal:true)
                        if let fname = asset.valueForKey("filename") as? String{
                            newItemImage.fileName = fname
                        }
                        newItemImage.thumbnail = result
                        asset.requestContentEditingInputWithOptions(PHContentEditingInputRequestOptions()) { (contentEditingInput, info) -> Void in

                            //Get full image
                            let url = contentEditingInput!.fullSizeImageURL

                            newItemImage.url = url
                        }
                        //DO SOMWTHING WITH IMAGE
                    }else
                    {
                        //DO SOMWTHING ELSE
                    }
            })
        }
    }

在这个阶段我有两个问题:

  1. 最重要的一个,但我想这是随之而来的。它需要太多的处理能力和从光盘读取。它在模拟器中运行良好。但是当我在具有 300-500 张图像的测试设备(iPhone 5、4S)上运行它时,它会冻结一两分钟,同时获取所有图像。我猜它会一一读取整个集合,调整它的大小并获取所有这些信息。有没有更简单的方法?就像使用 ALAsset 一样,我可以直接得到缩略图,它飞得很快。
  2. 由于某种原因,它做了两次工作。首先它给了我我想要的东西,小缩略图~ 60 * 40 或大约。一旦它给了我,几秒钟后它会进入另一个循环并开始检索更高分辨率的图像,比如 960 * 640 左右,这就是发生冻结的地方。我不确定为什么会这样,但确实如此。

对这两个问题的任何意见将不胜感激。

4

2 回答 2

2

由于某种原因,它做了两次工作

因为那是它应该做的。正如您所发现的,从照片库中获取图像需要时间。因此,默认行为是我们尽快提供低分辨率图像,以便您可以显示一些内容;然后我们再次调用,可能是几次,使用质量更好的图像版本。此外,获取是异步形成的。因此,完全有可能同时发生多个 fetch 请求,这可能会导致您的代码开始跌跌撞撞,正如您所发现的那样。

如果您不喜欢这样,请将 fetch 选项设置synchronoustrue- 但是您必须在后台队列上进行整个调用!通过在串行队列中执行此操作,您可以确保依次单独执行调用,并且每个图像将仅传递一次。此外,那时(而且只有那时)你的PHImageRequestOptions.DeliveryModeFastFormat意志才会被服从。在对收到的图像进行任何操作之前,不要忘记退回到主线程!

于 2015-09-21T02:45:13.990 回答
2

你也可以使用我写的这个扩展:

import Photos

extension PHAsset {
    var thumbnailImage : UIImage {
        get {
            let manager = PHImageManager.default()
            let option = PHImageRequestOptions()
            var thumbnail = UIImage()
            option.isSynchronous = true
            manager.requestImage(for: self, targetSize: CGSize(width: 300, height: 300), contentMode: .aspectFit, options: option, resultHandler: {(result, info)->Void in
                thumbnail = result!
            })
            return thumbnail
        }
    }
}
于 2018-02-12T21:09:00.310 回答