在Swift 4中将属性观察器添加到Decodable协议

时间:2017-10-20 03:30:29

标签: 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
}

does not conform to protocol 'Decodable'

其他人有解决方案吗?

1 个答案:

答案 0 :(得分:1)

下面我将列举一些处理这种情况的方法,但正确的答案是你不应该在解析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
    }
    

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

  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中的手动编码和解码

  3. 我提供了以上两种模式,作为如何使某些属性不属于JSON但以其他方式初始化的示例。但是因为你使用Data(contentsOf:),所以这些都不是真的可取。但是,如果有问题的财产不需要花费一些时间来完成同步任务,那么他们应该注意好的模式,就像你在这里一样。

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

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

      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
          }
      }
      
    2. 这些是你可以采用的几种方法。