swift - 在 Swift 4 中将属性观察器添加到 Decodable 协议(protocol)

标签 swift decodable

我正在从 JSON 解析中检索 URL,执行此操作后,我希望将该 URL 转换为 UIImage。这是我当前的数据结构:

struct Book: Decodable {
    let author: String
    let artworkURL: URL
    let genres: [Genre]
    let name: String
    let releaseDate: String
}

我尝试这样做:

struct Book: Decodable {
    let author: String
    var bookArtwork: UIImage
    let artworkURL: URL {
    didSet {
        do {
            if let imageData: Data = try Data(contentsOf: artworkURL) {
                bookArtwork = UIImage(data: imageData)!
            }
        } catch {
            print(error)
        }
    }
    let genres: [Genre]
    let name: String
    let releaseDate: String
}

但是不符合“Decodable”协议(protocol)

还有人有解决办法吗?

最佳答案

下面我将列举一些处理此类场景的方法,但正确的答案是您不应该在解析 JSON 期间检索图像。而且您绝对不应该同步执行此操作(这就是 Data(contentsOf:) 所做的)。

相反,您应该只检索 UI 需要的图像。您希望将图像检索到可以在内存不足的情况下清除的内容,以响应 .UIApplicationDidReceiveMemoryWarning 系统通知。最重要的是,如果您不小心并且几乎总是希望将图像与模型对象本身解耦,则图像可能会占用惊人的内存量。


但是,如果您绝对决定将图像合并到 Book 对象中(我再次建议不要这样做),您可以:

  1. 您可以将 bookArtwork 设为 lazy 变量:

    struct Book: Decodable {
        let author: String
        lazy var bookArtwork: UIImage = {
            let data = try! Data(contentsOf: artworkURL)
            return UIImage(data: data)!
        }()
        let artworkURL: URL
        let genres: [Genre]
        let name: String
        let releaseDate: String
    }
    

    在这种情况下,这是一种可怕的模式(同样,因为我们永远不应该进行同步网络调用),但它说明了惰性变量的想法。有时您也会使用计算属性来执行这种模式。

  2. 同样糟糕(出于同样的原因,同步网络调用是邪恶的),您还可以实现自定义 init 方法:

    struct Book: Decodable {
        let author: String
        let bookArtwork: UIImage
        let artworkURL: URL
        let genres: [Genre]
        let name: String
        let releaseDate: String
    
        init(from decoder: Decoder) throws {
            let values = try decoder.container(keyedBy: CodingKeys.self)
            author = try values.decode(String.self, forKey: .author)
            artworkURL = try values.decode(URL.self, forKey: .artworkURL)
            genres = try values.decode([Genre].self, forKey: .genres)
            name = try values.decode(String.self, forKey: .name)
            releaseDate = try values.decode(String.self, forKey: .releaseDate)
    
            // special processing for artworkURL
            let data = try Data(contentsOf: artworkURL)
            bookArtwork = UIImage(data: data)!
        }
    
        enum CodingKeys: String, CodingKey {
            case author, artworkURL, genres, name, releaseDate
        }
    }
    

    事实上,这比之前的模式更糟糕,因为至少使用第一个选项时,同步网络调用会被推迟到您引用 UIImage 属性时。

    但有关此一般模式的更多信息,请参阅 Encoding and Decoding Custom Types 中的手动编码和解码 .

我提供了上述两种模式,作为示例,说明如何拥有一些不属于 JSON 的属性,但以其他方式初始化。但因为您使用的是 Data(contentsOf:),所以这些都不是真正可取的。但它们是需要注意的好模式,以防相关属性不需要一些耗时的同步任务,就像您在此处所做的那样。

在这种情况下,我认为最简单的方法是提供一种在需要时异步检索图像的方法:

  • 完全消除同步网络调用,只提供异步方法来检索图像:

    struct Book: Decodable {
        let author: String
        let artworkURL: URL
        let genres: [Genre]
        let name: String
        let releaseDate: String
    
        func retrieveImage(completionHandler: @escaping (UIImage?, Error?) -> Void) {
            let task = URLSession.shared.dataTask(with: artworkURL) { data, _, error in
                guard let data = data, error == nil else {
                    completionHandler(nil, error)
                    return
                }
                completionHandler(UIImage(data: data), nil)
            }
            task.resume()
        }
    }
    

    然后,当您的 UI 需要图像时,您可以延迟检索它:

    book.retrieveImage { image, error in
        DispatchQueue.main.async { image, error in
           cell.imageView.image = image
        }
    }
    
  • 这些是您可以采用的一些方法。

    关于swift - 在 Swift 4 中将属性观察器添加到 Decodable 协议(protocol),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46841999/

    相关文章:

    ios - AudioKit 后台音频电池使用

    ios - 在继续循环之前让后台任务完成

    ios - "hidden"与#keyPath(UIView.isHidden)

    json - 使用 Swift Decodable 解析任意格式的 JSON 日期

    swift - 如何在 Swift 中销毁单例

    swift - 在 Swift Sprite Kit 中预加载纹理时如何防止白屏?

    json - 如何快速将这个异构对象与模型映射?

    swift - 使用 Decodable 构造嵌套数据

    json - 处理包含多种类型的 JSON 数组 - Swift 4 Decodable

    json - Swift 4 用 ' - ' 字母解码 json