在 Swift 4 中将 属性 观察者添加到 Decodable 协议
Adding property observer to Decodable protocol in Swift 4
我正在从 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'
还有其他人有解决方案吗?
下面我将列举几种处理这种情况的方法,但正确的答案是你根本不应该在 JSON 的解析过程中检索图像。而且您绝对不应该同步执行此操作(Data(contentsOf:)
就是这样做的)。
相反,您应该只检索 UI 需要的图像。并且您想将图像检索到可以在内存不足的情况下清除的内容,以响应 .UIApplicationDidReceiveMemoryWarning
系统通知。底线是,如果您不小心并且您几乎总是想将图像与模型对象本身分离,图像可能会占用大量内存。
但是,如果您绝对决定将图像合并到 Book
对象中(我再次建议您不要这样做),您可以:
您可以将 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
变量的想法。您有时也会使用计算属性来执行这种模式。
同样糟糕(出于同样的原因,同步网络调用是邪恶的),您还可以实现自定义 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
属性.[=24 时=]
但有关此一般模式的更多信息,请参阅 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
}
}
这些是您可以采用的一些方法。
我正在从 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'
还有其他人有解决方案吗?
下面我将列举几种处理这种情况的方法,但正确的答案是你根本不应该在 JSON 的解析过程中检索图像。而且您绝对不应该同步执行此操作(Data(contentsOf:)
就是这样做的)。
相反,您应该只检索 UI 需要的图像。并且您想将图像检索到可以在内存不足的情况下清除的内容,以响应 .UIApplicationDidReceiveMemoryWarning
系统通知。底线是,如果您不小心并且您几乎总是想将图像与模型对象本身分离,图像可能会占用大量内存。
但是,如果您绝对决定将图像合并到 Book
对象中(我再次建议您不要这样做),您可以:
您可以将
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
变量的想法。您有时也会使用计算属性来执行这种模式。同样糟糕(出于同样的原因,同步网络调用是邪恶的),您还可以实现自定义
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
属性.[=24 时=]但有关此一般模式的更多信息,请参阅 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 } }
这些是您可以采用的一些方法。